diff --git a/pom.xml b/pom.xml index 58acc2f9..4f087cf5 100644 --- a/pom.xml +++ b/pom.xml @@ -245,21 +245,28 @@ 3.3.3 - + + + com.github.wnameless.json + json-flattener + 0.16.4 + + + cn.dev33 sa-token-core 1.37.0 - + cn.dev33 sa-token-spring-boot3-starter 1.37.0 - + cn.dev33 sa-token-redis-jackson @@ -343,25 +350,18 @@ 3.1.944 - - - com.aliyun - dysmsapi20170525 - 2.0.24 - - - - - com.tencentcloudapi - tencentcloud-sdk-java-sms - 3.1.893 - - org.dromara.sms4j sms4j-javase-plugin - 3.1.1 + 3.3.4 + + + + + org.dromara.sms4j + sms4j-oa-core + 3.3.4 @@ -371,6 +371,14 @@ 6.4.11 + + + org.junit.jupiter + junit-jupiter-api + 5.10.1 + test + + com.baomidou @@ -432,7 +440,7 @@ org.apache.maven.plugins maven-compiler-plugin - 3.12.1 + 3.7.0 -parameters @@ -461,11 +469,19 @@ src/main/resources + + _sql/* + *.md + src/main/java **/*.xml + **/*.ttf + **/*.ttc + **/*.TTF + **/*.TTC diff --git a/snowy-common/pom.xml b/snowy-common/pom.xml index 6c0d25d5..963cf0ac 100644 --- a/snowy-common/pom.xml +++ b/snowy-common/pom.xml @@ -129,5 +129,11 @@ com.alibaba easyexcel + + + + com.github.wnameless.json + json-flattener + diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/cache/CommonCacheOperator.java b/snowy-common/src/main/java/vip/xiaonuo/common/cache/CommonCacheOperator.java index 0c3e784a..600b4011 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/cache/CommonCacheOperator.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/cache/CommonCacheOperator.java @@ -58,21 +58,13 @@ public class CommonCacheOperator { public Collection getAllKeys() { Set keys = redisTemplate.keys(CACHE_KEY_PREFIX + "*"); - if (keys != null) { - // 去掉缓存key的common prefix前缀 - return keys.stream().map(key -> StrUtil.removePrefix(key, CACHE_KEY_PREFIX)).collect(Collectors.toSet()); - } else { - return CollectionUtil.newHashSet(); - } + // 去掉缓存key的common prefix前缀 + return keys.stream().map(key -> StrUtil.removePrefix(key, CACHE_KEY_PREFIX)).collect(Collectors.toSet()); } public Collection getAllValues() { Set keys = redisTemplate.keys(CACHE_KEY_PREFIX + "*"); - if (keys != null) { - return redisTemplate.opsForValue().multiGet(keys); - } else { - return CollectionUtil.newArrayList(); - } + return redisTemplate.opsForValue().multiGet(keys); } public Map getAllKeyValues() { @@ -86,8 +78,6 @@ public class CommonCacheOperator { public void removeBatch(String pattern) { Set keys = redisTemplate.keys(CACHE_KEY_PREFIX + pattern); - if (keys != null) { - redisTemplate.delete(keys); - } + redisTemplate.delete(keys); } } diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/consts/CacheConstant.java b/snowy-common/src/main/java/vip/xiaonuo/common/consts/CacheConstant.java index 44cdf430..ffcde459 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/consts/CacheConstant.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/consts/CacheConstant.java @@ -13,7 +13,8 @@ package vip.xiaonuo.common.consts; /** - * @description 缓存静态常量 + * 缓存静态常量 + * * @author dongxiayu * @date 2023/1/30 0:44 **/ diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/enums/CommonSortOrderEnum.java b/snowy-common/src/main/java/vip/xiaonuo/common/enums/CommonSortOrderEnum.java index 30d7cd53..74f04ea7 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/enums/CommonSortOrderEnum.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/enums/CommonSortOrderEnum.java @@ -42,8 +42,4 @@ public enum CommonSortOrderEnum { throw new CommonException("不支持该排序方式:{}", value); } } - - public String getValue() { - return value.toLowerCase(); - } } diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/handler/CommonSm4CbcTypeHandler.java b/snowy-common/src/main/java/vip/xiaonuo/common/handler/CommonSm4CbcTypeHandler.java index 0b70a808..22b51a15 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/handler/CommonSm4CbcTypeHandler.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/handler/CommonSm4CbcTypeHandler.java @@ -42,20 +42,20 @@ public class CommonSm4CbcTypeHandler extends BaseTypeHandler { public T getNullableResult(ResultSet rs, String columnName) throws SQLException { String columnValue = rs.getString(columnName); //有一些可能是空字符 - return StringUtils.isBlank(columnValue) ? (T)columnValue : (T)CommonCryptogramUtil.doSm4CbcDecrypt(columnValue); + return StringUtils.isBlank(columnValue) ? (T)columnValue : (T) CommonCryptogramUtil.doSm4CbcDecrypt(columnValue); } @SuppressWarnings("ALL") @Override public T getNullableResult(ResultSet rs, int columnIndex) throws SQLException { String columnValue = rs.getString(columnIndex); - return StringUtils.isBlank(columnValue) ? (T)columnValue : (T)CommonCryptogramUtil.doSm4CbcDecrypt(columnValue); + return StringUtils.isBlank(columnValue) ? (T)columnValue : (T) CommonCryptogramUtil.doSm4CbcDecrypt(columnValue); } @SuppressWarnings("ALL") @Override public T getNullableResult(CallableStatement cs, int columnIndex) throws SQLException { String columnValue = cs.getString(columnIndex); - return StringUtils.isBlank(columnValue) ? (T)columnValue : (T)CommonCryptogramUtil.doSm4CbcDecrypt(columnValue); + return StringUtils.isBlank(columnValue) ? (T)columnValue : (T) CommonCryptogramUtil.doSm4CbcDecrypt(columnValue); } } diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/listener/CommonDataChangeEventCenter.java b/snowy-common/src/main/java/vip/xiaonuo/common/listener/CommonDataChangeEventCenter.java index 531b93fd..ef29901d 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/listener/CommonDataChangeEventCenter.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/listener/CommonDataChangeEventCenter.java @@ -15,6 +15,7 @@ package vip.xiaonuo.common.listener; import cn.hutool.json.JSONArray; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; +import lombok.Getter; import vip.xiaonuo.common.exception.CommonException; import java.util.ArrayList; @@ -31,15 +32,11 @@ public class CommonDataChangeEventCenter { // --------- 注册侦听器 - private static List listenerList = new ArrayList<>(); - /** * 获取已注册的所有侦听器 - * @return / */ - public static List getListenerList() { - return listenerList; - } + @Getter + private static List listenerList = new ArrayList<>(); /** * 重置侦听器集合 diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/page/CommonPageRequest.java b/snowy-common/src/main/java/vip/xiaonuo/common/page/CommonPageRequest.java index d3728782..cff1c304 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/page/CommonPageRequest.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/page/CommonPageRequest.java @@ -56,7 +56,6 @@ public class CommonPageRequest { } } catch (Exception e) { log.error(">>> 分页条数转换异常:", e); - size = 20; } } @@ -67,7 +66,6 @@ public class CommonPageRequest { page = Convert.toInt(pageString); } catch (Exception e) { log.error(">>> 分页页数转换异常:", e); - page = 1; } } Page objectPage = new Page<>(page, size); diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/pojo/CommonResult.java b/snowy-common/src/main/java/vip/xiaonuo/common/pojo/CommonResult.java index ffa97398..1061a6c3 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/pojo/CommonResult.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/pojo/CommonResult.java @@ -13,7 +13,9 @@ package vip.xiaonuo.common.pojo; import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import java.io.Serial; import java.io.Serializable; /** @@ -23,6 +25,7 @@ import java.io.Serializable; * @date 2022/8/15 16:08 **/ public class CommonResult implements Serializable{ + @Serial private static final long serialVersionUID = 1L; public static final int CODE_SUCCESS = 200; public static final int CODE_ERROR = 500; @@ -30,9 +33,17 @@ public class CommonResult implements Serializable{ @Schema(description = "状态码") private int code; + /** + * 获取msg + */ + @Getter @Schema(description = "提示语") private String msg; + /** + * 获取data + */ + @Getter @Schema(description = "返回数据") private T data; @@ -53,21 +64,6 @@ public class CommonResult implements Serializable{ return this.code; } - /** - * 获取msg - * @return msg - */ - public String getMsg() { - return this.msg; - } - /** - * 获取data - * @return data - */ - public T getData() { - return this.data; - } - /** * 给code赋值,连缀风格 * @param code code diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/prop/CommonProperties.java b/snowy-common/src/main/java/vip/xiaonuo/common/prop/CommonProperties.java index a54b2b29..a5b845e1 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/prop/CommonProperties.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/prop/CommonProperties.java @@ -29,9 +29,6 @@ import org.springframework.stereotype.Component; @ConfigurationProperties(prefix = "snowy.config.common") public class CommonProperties { - /** 前端地址 */ - private String frontUrl; - /** 后端地址 */ private String backendUrl; } diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonCryptogramUtil.java b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonCryptogramUtil.java index 53c9bf80..2a2560f0 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonCryptogramUtil.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonCryptogramUtil.java @@ -19,7 +19,7 @@ import com.antherd.smcrypto.sm4.Sm4Options; import lombok.extern.slf4j.Slf4j; /** - * 加密工具类,本框架目前使用 https://github.com/antherd/sm-crypto 项目中一些加解密方式 + * 加密工具类,本框架目前使用 sm-crypto 项目中一些加解密方式 * 使用小伙伴需要过等保密评相关,请在此处更改为自己的加密方法,或加密机,使用加密机同时需要替换公钥,私钥在内部无法导出,提供加密的方法 * 如果不涉及到加密机方面的内容,请更改公私要为自己重新生成的,生成方式请看集成的sm-crypto主页 * diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonEmailUtil.java b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonEmailUtil.java index c087c585..a3effdfb 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonEmailUtil.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonEmailUtil.java @@ -30,8 +30,8 @@ public class CommonEmailUtil { * @author xuyuxiang * @date 2022/8/15 13:32 **/ - public static boolean isEmail(String email) { - return Validator.isEmail(email); + public static boolean isNotEmail(String email) { + return !Validator.isEmail(email); } /** @@ -42,7 +42,7 @@ public class CommonEmailUtil { **/ public static void validEmail(String emails) { StrUtil.split(emails, StrUtil.COMMA).forEach(email -> { - if(!isEmail(email)) { + if(isNotEmail(email)) { throw new CommonException("邮件地址:{}格式错误", email); } }); diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonIpAddressUtil.java b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonIpAddressUtil.java index 159e5c80..2d82fded 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonIpAddressUtil.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonIpAddressUtil.java @@ -26,7 +26,7 @@ import java.io.InputStream; /** * 根据ip地址定位工具类,离线方式 - * 参考地址:https://gitee.com/lionsoul/ip2region/tree/master/binding/java + * 参考地址:ip2region * * @author xuyuxiang * @date 2020/3/16 11:25 diff --git a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonTimeFormatUtil.java b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonTimeFormatUtil.java index c3f0f294..d2abf21e 100644 --- a/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonTimeFormatUtil.java +++ b/snowy-common/src/main/java/vip/xiaonuo/common/util/CommonTimeFormatUtil.java @@ -77,29 +77,15 @@ public class CommonTimeFormatUtil { String weekday; //获取是本周的第几天 int dayOfWeek = DateUtil.dayOfWeek(date) - 1; - switch (dayOfWeek) { - case 1: - weekday = "周一"; - break; - case 2: - weekday = "周二"; - break; - case 3: - weekday = "周三"; - break; - case 4: - weekday = "周四"; - break; - case 5: - weekday = "周五"; - break; - case 6: - weekday = "周六"; - break; - default: - weekday = "周日"; - break; - } + weekday = switch (dayOfWeek) { + case 1 -> "周一"; + case 2 -> "周二"; + case 3 -> "周三"; + case 4 -> "周四"; + case 5 -> "周五"; + case 6 -> "周六"; + default -> "周日"; + }; //显示本周时分 return weekday + " " + DateUtil.format(date, "HH:mm"); } else { diff --git a/snowy-plugin-api/snowy-plugin-auth-api/pom.xml b/snowy-plugin-api/snowy-plugin-auth-api/pom.xml index 649c8cd4..d5aceb36 100644 --- a/snowy-plugin-api/snowy-plugin-auth-api/pom.xml +++ b/snowy-plugin-api/snowy-plugin-auth-api/pom.xml @@ -27,4 +27,4 @@ sa-token-core - \ No newline at end of file + diff --git a/snowy-plugin-api/snowy-plugin-auth-api/src/main/java/vip/xiaonuo/auth/api/SaBaseLoginUserApi.java b/snowy-plugin-api/snowy-plugin-auth-api/src/main/java/vip/xiaonuo/auth/api/SaBaseLoginUserApi.java index 151f384f..fb2f1174 100644 --- a/snowy-plugin-api/snowy-plugin-auth-api/src/main/java/vip/xiaonuo/auth/api/SaBaseLoginUserApi.java +++ b/snowy-plugin-api/snowy-plugin-auth-api/src/main/java/vip/xiaonuo/auth/api/SaBaseLoginUserApi.java @@ -66,6 +66,14 @@ public interface SaBaseLoginUserApi { **/ SaBaseLoginUser getUserByPhone(String phone); + /** + * 根据邮箱获取B端用户信息,查不到则返回null + * + * @author xuyuxiang + * @date 2022/3/10 16:14 + **/ + SaBaseLoginUser getUserByEmail(String email); + /** * 根据手机号获取C端用户信息,查不到则返回null * @@ -74,6 +82,14 @@ public interface SaBaseLoginUserApi { **/ SaBaseClientLoginUser getClientUserByPhone(String phone); + /** + * 根据邮箱获取C端用户信息,查不到则返回null + * + * @author xuyuxiang + * @date 2022/3/10 16:14 + **/ + SaBaseClientLoginUser getClientUserByEmail(String email); + /** * 根据用户id获取用户集合 * @@ -121,4 +137,44 @@ public interface SaBaseLoginUserApi { * @date 2022/4/27 22:57 */ void updateUserLoginInfo(String userId, String device); + + /** + * 使用手机号创建B端用户 + * + * @author xuyuxiang + * @date 2022/3/10 16:14 + **/ + SaBaseLoginUser createUserWithPhone(String phone); + + /** + * 使用手机号创建C端用户 + * + * @author xuyuxiang + * @date 2022/3/10 16:14 + **/ + SaBaseClientLoginUser createClientUserWithPhone(String phone); + + /** + * 使用邮箱创建B端用户 + * + * @author xuyuxiang + * @date 2022/3/10 16:14 + **/ + SaBaseLoginUser createUserWithEmail(String email); + + /** + * 使用邮箱创建C端用户 + * + * @author xuyuxiang + * @date 2022/3/10 16:14 + **/ + SaBaseClientLoginUser createClientUserWithEmail(String email); + + /** + * 执行注册 + * + * @author xuyuxiang + * @date 2022/3/10 16:14 + **/ + void doRegister(String account, String password); } diff --git a/snowy-plugin-api/snowy-plugin-dev-api/pom.xml b/snowy-plugin-api/snowy-plugin-dev-api/pom.xml index c8155fed..bd10263b 100644 --- a/snowy-plugin-api/snowy-plugin-dev-api/pom.xml +++ b/snowy-plugin-api/snowy-plugin-dev-api/pom.xml @@ -63,28 +63,22 @@ tencentcloud-sdk-java-ses - - - com.aliyun - dysmsapi20170525 - - - - - com.tencentcloudapi - tencentcloud-sdk-java-sms - - org.dromara.sms4j sms4j-javase-plugin + + + org.dromara.sms4j + sms4j-oa-core + + com.github.oshi oshi-core - \ No newline at end of file + diff --git a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevEmailApi.java b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevEmailApi.java index 0704bb7d..9d5a7359 100644 --- a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevEmailApi.java +++ b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevEmailApi.java @@ -27,6 +27,28 @@ import java.util.Map; **/ public interface DevEmailApi { + /** + * 动态发送TXT邮件(使用系统配置的默认邮件引擎) + * + * @param tos 收件人邮箱,逗号拼接 + * @param subject 邮件主题 + * @param content 邮件内容 + * @author xuyuxiang + * @date 2022/2/7 22:29 + */ + void sendDynamicTxtEmail(String tos, String subject, String content); + + /** + * 动态发送HTML邮件(使用系统配置的默认邮件引擎) + * + * @param tos 收件人邮箱,逗号拼接 + * @param subject 邮件主题 + * @param content 邮件内容 + * @author xuyuxiang + * @date 2022/2/7 22:29 + */ + void sendDynamicHtmlEmail(String tos, String subject, String content); + /* =========本地邮件========= */ /** diff --git a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevFileApi.java b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevFileApi.java index e0a20d28..21ae3487 100644 --- a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevFileApi.java +++ b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevFileApi.java @@ -31,6 +31,14 @@ public interface DevFileApi { **/ String uploadDynamicReturnId(MultipartFile file); + /** + * 动态上传文件返回url(使用系统配置的默认文件引擎) + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + String uploadDynamicReturnUrl(MultipartFile file); + /* =========本地文件========= */ /** @@ -119,6 +127,7 @@ public interface DevFileApi { */ JSONObject getFileInfoById(String id); + /** * 根据文件id物理删除文件 * diff --git a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevPushApi.java b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevPushApi.java new file mode 100644 index 00000000..d0bb90d3 --- /dev/null +++ b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevPushApi.java @@ -0,0 +1,109 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.api; + +/** + * 邮件API接口 + * + * @author xuyuxiang + * @date 2022/6/22 15:21 + **/ +public interface DevPushApi { + + /** + * 动态推送消息(使用系统配置的默认消息推送引擎) + * + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushDynamicText(String content, boolean noticeAll); + + /** + * 推送消息——飞书TXT + * + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushFeiShuText(String content, boolean noticeAll); + + /** + * 推送消息——钉钉TXT + * + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @param phones 通知的用户手机号,英文逗号分割 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushDingTalkText(String content, boolean noticeAll, String phones); + + /** + * 推送消息——钉钉MARKDOWN + * + * @param title 标题 + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushDingTalkMarkdown(String title, String content, boolean noticeAll); + + /** + * 推送消息——钉钉LINK + * + * @param title 标题 + * @param content 内容 + * @param picUrl 封面图片地址 + * @param messageUrl 消息链接地址 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushDingTalkLink(String title, String content, String picUrl,String messageUrl); + + /** + * 推送消息——企业微信TXT + * + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @param phones 通知的用户手机号,英文逗号分割 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushWorkWechatText(String content, boolean noticeAll, String phones); + + /** + * 推送消息——企业微信MARKDOWN + * + * @param title 标题 + * @param content 内容 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushWorkWechatMarkdown(String title, String content); + + /** + * 推送消息——企业微信NEWS + * + * @param title 标题 + * @param content 内容 + * @param picUrl 封面图片地址 + * @param messageUrl 消息链接地址 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + void pushWorkWechatNews(String title, String content, String picUrl,String messageUrl); +} diff --git a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevSmsApi.java b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevSmsApi.java index 5d1523d4..7db035c9 100644 --- a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevSmsApi.java +++ b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevSmsApi.java @@ -12,6 +12,8 @@ */ package vip.xiaonuo.dev.api; +import cn.hutool.json.JSONObject; + /** * 短信API * @@ -20,6 +22,17 @@ package vip.xiaonuo.dev.api; **/ public interface DevSmsApi { + /** + * 动态发送短信(使用系统配置的默认短信引擎) + * + * @param phoneNumbers 手机号 + * @param templateCodeOrId 模板id或编码 + * @param paramMap 发送参数 + * @author xuyuxiang + * @date 2022/2/7 22:29 + */ + void sendDynamicSms(String phoneNumbers, String templateCodeOrId, JSONObject paramMap); + /* =========阿里云短信========= */ /** @@ -27,7 +40,7 @@ public interface DevSmsApi { * * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 * 上限为1000个手机号码。批量调用相对于单条调用及时性稍有延迟。 - * @param signName 短信服务控制台配置且审核通过的短信签名,为空则使用默认签名 + * @param signName 短信服务控制台配置且审核通过的短信签名 * @param templateCode 短信服务控制台配置且审核通过的模板编码 * @param templateParam 短信模板变量对应的实际值,JSON格式。支持传入多个参数,示例:{"name":"张三","number":"15038****76"} * @author xuyuxiang @@ -40,17 +53,15 @@ public interface DevSmsApi { /** * 发送短信 * - * @param sdkAppId 短信 SdkAppId,在 短信控制台 添加应用后生成的实际 SdkAppId,示例如1400006666。 - * 可前往 [短信控制台](https://console.cloud.tencent.com/smsv2/app-manage) 查看 * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 * 上限为1000个手机号码。批量调用相对于单条调用及时性稍有延迟。 - * @param signName 短信服务控制台配置且审核通过的短信签名,为空则使用默认签名 + * @param signName 短信服务控制台配置且审核通过的短信签名 * @param templateCode 短信服务控制台配置且审核通过的模板编码 * @param templateParam 短信模板变量对应的顺序。支持传入多个参数,逗号拼接,示例:"张三,15038****76,进行中"} * @author xuyuxiang * @date 2022/2/24 13:42 **/ - void sendSmsTencent(String sdkAppId, String phoneNumbers, String signName, String templateCode, String templateParam); + void sendSmsTencent(String phoneNumbers, String signName, String templateCode, String templateParam); /* =========小诺短信========= */ @@ -58,10 +69,11 @@ public interface DevSmsApi { * 发送短信 * * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 - * @param signName 短信签名,为空则使用默认签名 - * @param message 短信内容 - * @author xuyuxiang - * @date 2022/2/24 13:42 + * 上限为1000个手机号码。批量调用相对于单条调用及时性稍有延迟。 + * @param signName 短信服务控制台配置且审核通过的短信签名 + * @param message 短信内容,发送时编写好的整条短信内容,不带签名【】 + * @author yubaoshan + * @date 2024/5/20 12:00 **/ void sendSmsXiaonuo(String phoneNumbers, String signName, String message); } diff --git a/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevWeakPasswordApi.java b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevWeakPasswordApi.java new file mode 100644 index 00000000..f87155f8 --- /dev/null +++ b/snowy-plugin-api/snowy-plugin-dev-api/src/main/java/vip/xiaonuo/dev/api/DevWeakPasswordApi.java @@ -0,0 +1,32 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.api; + +import java.util.List; + +/** + * 弱密码库APi接口 + * + * @author xuyuxiang + * @date 2022/6/17 10:37 + **/ +public interface DevWeakPasswordApi { + + /** + * 获取弱密码库列表 + * + * @author xuyuxiang + * @date 2022/6/17 11:11 + **/ + List weakPasswordList(); +} diff --git a/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileApi.java b/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileApi.java new file mode 100644 index 00000000..7634d4bb --- /dev/null +++ b/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileApi.java @@ -0,0 +1,22 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.mobile.api; + +/** + * 移动端综合API + * + * @author xuyuxiang + * @date 2022/9/26 14:24 + **/ +public interface MobileApi { +} diff --git a/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileButtonApi.java b/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileButtonApi.java index 49e184df..7d71ed88 100644 --- a/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileButtonApi.java +++ b/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileButtonApi.java @@ -28,5 +28,5 @@ public interface MobileButtonApi { * @author 每天一点 * @date 2023/2/5 13:26 **/ - List listByIds(List idList); + List listButtonCodeListByIdList(List idList); } diff --git a/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileMenuApi.java b/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileMenuApi.java index 1636a883..54392b71 100644 --- a/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileMenuApi.java +++ b/snowy-plugin-api/snowy-plugin-mobile-api/src/main/java/vip/xiaonuo/mobile/api/MobileMenuApi.java @@ -33,6 +33,14 @@ public interface MobileMenuApi { **/ List mobileMenuTreeSelector(); + /** + * 获取移动端菜单授权树 + * + * @author xuyuxiang + * @date 2023/1/31 10:10 + **/ + List mobileMenuTreeSelector(List originDataList); + /** * 获取移动端登录菜单树 * diff --git a/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysApi.java b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysApi.java new file mode 100644 index 00000000..d5034865 --- /dev/null +++ b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysApi.java @@ -0,0 +1,30 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.api; + +/** + * 系统模块综合API + * + * @author xuyuxiang + * @date 2022/9/26 14:24 + **/ +public interface SysApi { + + /** + * 获取系统默认密码 + * + * @author xuyuxiang + * @date 2022/9/26 14:25 + **/ + String getDefaultPassword(); +} diff --git a/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysModuleApi.java b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysModuleApi.java index b53c0580..a760400a 100644 --- a/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysModuleApi.java +++ b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysModuleApi.java @@ -13,6 +13,7 @@ package vip.xiaonuo.sys.api; import cn.hutool.json.JSONObject; + import java.util.List; /** diff --git a/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysPositionApi.java b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysPositionApi.java index e4fea209..7cd782a6 100644 --- a/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysPositionApi.java +++ b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysPositionApi.java @@ -37,5 +37,5 @@ public interface SysPositionApi { * @author xuyuxiang * @date 2022/7/22 14:47 **/ - Page positionSelector(String orgId, String searchKey); + Page positionSelector(String orgId, String searchKey, Integer current, Integer size); } diff --git a/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysRoleApi.java b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysRoleApi.java index d39c4ffc..f691df51 100644 --- a/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysRoleApi.java +++ b/snowy-plugin-api/snowy-plugin-sys-api/src/main/java/vip/xiaonuo/sys/api/SysRoleApi.java @@ -48,4 +48,20 @@ public interface SysRoleApi { * @date 2022/11/1 15:58 **/ void grantForGenMenuAndButton(String menuId); + + /** + * 获取资源授权树 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + List resourceTreeSelector(); + + /** + * 获取权限授权树 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + List permissionTreeSelector(); } diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/config/AuthConfigure.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/config/AuthConfigure.java index 3d88eadb..957b4362 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/config/AuthConfigure.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/config/AuthConfigure.java @@ -17,7 +17,8 @@ import cn.dev33.satoken.interceptor.SaInterceptor; import cn.dev33.satoken.stp.StpInterface; import cn.dev33.satoken.stp.StpLogic; import cn.dev33.satoken.strategy.SaStrategy; -import cn.hutool.json.JSONUtil; +import cn.hutool.core.collection.CollectionUtil; +import cn.hutool.core.util.StrUtil; import jakarta.annotation.Resource; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; @@ -71,6 +72,7 @@ public class AuthConfigure implements WebMvcConfigurer { @Bean("stpClientLogic") public StpLogic getStpClientLogic(SaTokenConfig saTokenConfig) { + // 重写Sa-Token的StpLogic,默认客户端类型为C return new StpLogic(SaClientTypeEnum.C.getValue()).setConfig(saTokenConfig); } @@ -105,7 +107,12 @@ public class AuthConfigure implements WebMvcConfigurer { } else { permissionListObject = commonCacheOperator.get(CacheConstant.AUTH_C_PERMISSION_LIST_CACHE_KEY + loginId); } - return JSONUtil.parseArray(permissionListObject).toList(String.class); + // 转为字符串 + String permissionListString = permissionListObject.toString(); + // 去除首尾的方括号 + String trimmedStr = StrUtil.sub(permissionListString, 1, -1); + // 使用逗号和空格分割字符串,并转换为列表 + return CollectionUtil.newArrayList(StrUtil.split(trimmedStr, ", ")); } /** diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/util/AuthEmailFormatUtl.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/util/AuthEmailFormatUtl.java new file mode 100644 index 00000000..ced77820 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/util/AuthEmailFormatUtl.java @@ -0,0 +1,48 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.core.util; + +import cn.hutool.core.date.DateUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONObject; +import vip.xiaonuo.dev.api.DevConfigApi; + +/** + * 认证相关邮件格式化工具类 + * + * @author xuyuxiang + * @date 2025/3/21 19:07 + **/ +public class AuthEmailFormatUtl { + + /** 系统名称 */ + private static final String SNOWY_SYS_NAME_KEY = "SNOWY_SYS_NAME"; + + /** + * 格式化邮件内容 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static String format(String content, JSONObject paramMap) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 获取系统名称 + String sysName = devConfigApi.getValueByKey(SNOWY_SYS_NAME_KEY); + // 系统名称 + paramMap.set("sysName", sysName); + // 当前时间 + paramMap.set("sysNowTime", DateUtil.now()); + return StrUtil.format(content, paramMap); + } +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/util/AuthExceptionUtil.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/util/AuthExceptionUtil.java index 54d76038..9bf33693 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/util/AuthExceptionUtil.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/core/util/AuthExceptionUtil.java @@ -18,6 +18,12 @@ import lombok.extern.slf4j.Slf4j; import vip.xiaonuo.common.pojo.CommonResult; import vip.xiaonuo.common.util.CommonServletUtil; +/** + * 认证相关异常处理工具类 + * + * @author xuyuxiang + * @date 2021/10/9 14:24 + **/ @Slf4j public class AuthExceptionUtil { diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthClientController.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthClientController.java index de4ea207..5b20dbc5 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthClientController.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthClientController.java @@ -12,10 +12,11 @@ */ package vip.xiaonuo.auth.modular.login.controller; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; @@ -28,10 +29,13 @@ import vip.xiaonuo.auth.core.util.StpClientUtil; import vip.xiaonuo.auth.modular.login.param.AuthAccountPasswordLoginParam; import vip.xiaonuo.auth.modular.login.param.AuthGetPhoneValidCodeParam; import vip.xiaonuo.auth.modular.login.param.AuthPhoneValidCodeLoginParam; +import vip.xiaonuo.auth.modular.login.param.AuthRegisterParam; import vip.xiaonuo.auth.modular.login.result.AuthPicValidCodeResult; import vip.xiaonuo.auth.modular.login.service.AuthService; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; + /** * C端登录控制器 * @@ -39,6 +43,7 @@ import vip.xiaonuo.common.pojo.CommonResult; * @date 2021/12/23 21:50 */ @Tag(name = "C端登录控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 1) @RestController @Validated public class AuthClientController { @@ -52,6 +57,7 @@ public class AuthClientController { * @author xuyuxiang * @date 2022/7/8 9:26 **/ + @ApiOperationSupport(order = 1) @Operation(summary = "C端获取图片验证码") @GetMapping("/auth/c/getPicCaptcha") public CommonResult getPicCaptcha() { @@ -64,6 +70,7 @@ public class AuthClientController { * @author xuyuxiang * @date 2022/7/8 9:26 **/ + @ApiOperationSupport(order = 2) @Operation(summary = "C端获取手机验证码") @GetMapping("/auth/c/getPhoneValidCode") public CommonResult getPhoneValidCode(@Valid AuthGetPhoneValidCodeParam authGetPhoneValidCodeParam) { @@ -76,6 +83,7 @@ public class AuthClientController { * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 3) @Operation(summary = "C端账号密码登录") @PostMapping("/auth/c/doLogin") public CommonResult doLogin(@RequestBody @Valid AuthAccountPasswordLoginParam authAccountPasswordLoginParam) { @@ -88,6 +96,7 @@ public class AuthClientController { * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 4) @Operation(summary = "C端手机验证码登录") @PostMapping("/auth/c/doLoginByPhone") public CommonResult doLoginByPhone(@RequestBody @Valid AuthPhoneValidCodeLoginParam authPhoneValidCodeLoginParam) { @@ -100,6 +109,7 @@ public class AuthClientController { * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 5) @Operation(summary = "C端退出") @SaClientCheckLogin @GetMapping("/auth/c/doLogout") @@ -114,10 +124,25 @@ public class AuthClientController { * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 6) @Operation(summary = "C端获取用户信息") @SaClientCheckLogin @GetMapping("/auth/c/getLoginUser") public CommonResult getLoginUser() { return CommonResult.data(authService.getClientLoginUser()); } + + /** + * C端注册 + * + * @author xuyuxiang + * @date 2021/10/15 13:12 + **/ + @ApiOperationSupport(order = 7) + @Operation(summary = "C端注册") + @PostMapping("/auth/c/register") + public CommonResult register(@RequestBody @Valid AuthRegisterParam authRegisterParam) { + authService.register(authRegisterParam, SaClientTypeEnum.C.getValue()); + return CommonResult.ok(); + } } diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthController.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthController.java index 9802812b..9a8287fe 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthController.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/controller/AuthController.java @@ -14,10 +14,11 @@ package vip.xiaonuo.auth.modular.login.controller; import cn.dev33.satoken.annotation.SaCheckLogin; import cn.dev33.satoken.stp.StpUtil; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; @@ -25,13 +26,13 @@ import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RestController; import vip.xiaonuo.auth.core.enums.SaClientTypeEnum; import vip.xiaonuo.auth.core.pojo.SaBaseLoginUser; -import vip.xiaonuo.auth.modular.login.param.AuthAccountPasswordLoginParam; -import vip.xiaonuo.auth.modular.login.param.AuthGetPhoneValidCodeParam; -import vip.xiaonuo.auth.modular.login.param.AuthPhoneValidCodeLoginParam; +import vip.xiaonuo.auth.modular.login.param.*; import vip.xiaonuo.auth.modular.login.result.AuthPicValidCodeResult; import vip.xiaonuo.auth.modular.login.service.AuthService; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; + /** * B端登录控制器 * @@ -39,6 +40,7 @@ import vip.xiaonuo.common.pojo.CommonResult; * @date 2021/12/23 21:50 */ @Tag(name = "B端登录控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 2) @RestController @Validated public class AuthController { @@ -52,6 +54,7 @@ public class AuthController { * @author xuyuxiang * @date 2022/7/8 9:26 **/ + @ApiOperationSupport(order = 1) @Operation(summary = "B端获取图片验证码") @GetMapping("/auth/b/getPicCaptcha") public CommonResult getPicCaptcha() { @@ -59,23 +62,38 @@ public class AuthController { } /** - * B端获取手机验证码 + * B端获取手机登录验证码 * * @author xuyuxiang * @date 2022/7/8 9:26 **/ - @Operation(summary = "B端获取手机验证码") + @ApiOperationSupport(order = 2) + @Operation(summary = "B端获取手机登录验证码") @GetMapping("/auth/b/getPhoneValidCode") public CommonResult getPhoneValidCode(@Valid AuthGetPhoneValidCodeParam authGetPhoneValidCodeParam) { return CommonResult.data(authService.getPhoneValidCode(authGetPhoneValidCodeParam, SaClientTypeEnum.B.getValue())); } + /** + * B端获取邮箱登录验证码 + * + * @author xuyuxiang + * @date 2022/7/8 9:26 + **/ + @ApiOperationSupport(order = 3) + @Operation(summary = "B端获取邮箱登录验证码") + @GetMapping("/auth/b/getEmailValidCode") + public CommonResult getEmailValidCode(@Valid AuthGetEmailValidCodeParam authGetEmailValidCodeParam) { + return CommonResult.data(authService.getEmailValidCode(authGetEmailValidCodeParam, SaClientTypeEnum.B.getValue())); + } + /** * B端账号密码登录 * * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 4) @Operation(summary = "B端账号密码登录") @PostMapping("/auth/b/doLogin") public CommonResult doLogin(@RequestBody @Valid AuthAccountPasswordLoginParam authAccountPasswordLoginParam) { @@ -88,18 +106,33 @@ public class AuthController { * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 5) @Operation(summary = "B端手机验证码登录") @PostMapping("/auth/b/doLoginByPhone") public CommonResult doLoginByPhone(@RequestBody @Valid AuthPhoneValidCodeLoginParam authPhoneValidCodeLoginParam) { return CommonResult.data(authService.doLoginByPhone(authPhoneValidCodeLoginParam, SaClientTypeEnum.B.getValue())); } + /** + * B端邮箱验证码登录 + * + * @author xuyuxiang + * @date 2021/10/15 13:12 + **/ + @ApiOperationSupport(order = 6) + @Operation(summary = "B端邮箱验证码登录") + @PostMapping("/auth/b/doLoginByEmail") + public CommonResult doLoginByEmail(@RequestBody @Valid AuthEmailValidCodeLoginParam authEmailValidCodeLoginParam) { + return CommonResult.data(authService.doLoginByEmail(authEmailValidCodeLoginParam, SaClientTypeEnum.B.getValue())); + } + /** * B端退出 * * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 7) @Operation(summary = "B端退出") @SaCheckLogin @GetMapping("/auth/b/doLogout") @@ -114,10 +147,25 @@ public class AuthController { * @author xuyuxiang * @date 2021/10/15 13:12 **/ + @ApiOperationSupport(order = 8) @Operation(summary = "B端获取用户信息") @SaCheckLogin @GetMapping("/auth/b/getLoginUser") public CommonResult getLoginUser() { return CommonResult.data(authService.getLoginUser()); } + + /** + * B端注册 + * + * @author xuyuxiang + * @date 2021/10/15 13:12 + **/ + @ApiOperationSupport(order = 9) + @Operation(summary = "B端注册") + @PostMapping("/auth/b/register") + public CommonResult register(@RequestBody @Valid AuthRegisterParam authRegisterParam) { + authService.register(authRegisterParam, SaClientTypeEnum.B.getValue()); + return CommonResult.ok(); + } } diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthDeviceTypeEnum.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthDeviceTypeEnum.java index 0815ff2a..95065da8 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthDeviceTypeEnum.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthDeviceTypeEnum.java @@ -24,19 +24,13 @@ import vip.xiaonuo.common.exception.CommonException; @Getter public enum AuthDeviceTypeEnum { - /** - * PC端 - */ + /** PC端 */ PC("PC"), - /** - * 移动端 - */ + /** 移动端 */ APP("APP"), - /** - * 小程序端 - */ + /** 小程序端 */ MINI("MINI"); private final String value; diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthExceptionEnum.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthExceptionEnum.java index dfbfc913..7753b20f 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthExceptionEnum.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthExceptionEnum.java @@ -23,59 +23,37 @@ import lombok.Getter; @Getter public enum AuthExceptionEnum { - /** - * 验证码不能为空 - */ + /** 验证码不能为空 */ VALID_CODE_EMPTY("验证码不能为空"), - /** - * 验证码请求号不能为空 - */ + /** 验证码请求号不能为空 */ VALID_CODE_REQ_NO_EMPTY("验证码请求号不能为空"), - /** - * 验证码错误 - */ + /** 验证码错误 */ VALID_CODE_ERROR("验证码错误"), - /** - * 账号错误 - */ + /** 账号错误 */ ACCOUNT_ERROR("账号错误"), - /** - * 账号已停用 - */ + /** 账号已停用 */ ACCOUNT_DISABLED("账号已停用"), - /** - * 密码错误 - */ + /** 密码错误 */ PWD_ERROR("密码错误"), - /** - * 手机号格式错误 - */ + /** 手机号格式错误 */ PHONE_FORMAT_ERROR("手机号格式错误"), - /** - * 手机号不存在 - */ - PHONE_ERROR("手机号不存在"), + /** 邮箱格式错误 */ + EMAIL_FORMAT_ERROR("邮箱格式错误"), - /** - * 客户端类型不能为空 - */ + /** 客户端类型不能为空 */ CLIENT_TYPE_EMPTY("客户端类型不能为空"), - /** - * 客户端类型错误 - */ + /** 客户端类型错误 */ CLIENT_TYPE_ERROR("客户端类型错误"), - /** - * 密码解密失败,请检查前端公钥 - */ + /** 密码解密失败,请检查前端公钥 */ PWD_DECRYPT_ERROR("密码解密失败,请检查前端公钥"); private final String value; diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthPhoneOrEmailTypeEnum.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthPhoneOrEmailTypeEnum.java new file mode 100644 index 00000000..9030fd83 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthPhoneOrEmailTypeEnum.java @@ -0,0 +1,45 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.login.enums; + +import lombok.Getter; +import vip.xiaonuo.common.exception.CommonException; + +/** + * 手机号邮箱类型枚举 + * + * @author xuyuxiang + * @date 2021/10/11 14:02 + **/ +@Getter +public enum AuthPhoneOrEmailTypeEnum { + + /** 手机号 */ + PHONE("PHONE"), + + /** 邮箱 */ + EMAIL("EMAIL"); + + private final String value; + + AuthPhoneOrEmailTypeEnum(String value) { + this.value = value; + } + + public static void validate(String value) { + boolean flag = PHONE.getValue().equals(value) || EMAIL.getValue().equals(value); + if(!flag) { + throw new CommonException("不支持的手机号邮箱类型:{}", value); + } + } +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthSmsEngineTypeEnum.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthSmsEngineTypeEnum.java new file mode 100644 index 00000000..493e6fb6 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthSmsEngineTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.login.enums; + +import lombok.Getter; + +/** + * 短信发送引擎类型枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum AuthSmsEngineTypeEnum { + + /** 阿里云 */ + ALIYUN("ALIYUN"), + + /** 腾讯云 */ + TENCENT("TENCENT"), + + /** 小诺短信 */ + XIAONUO("XIAONUO"); + + private final String value; + + AuthSmsEngineTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthStrategyWhenNoUserWithPhoneOrEmailEnum.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthStrategyWhenNoUserWithPhoneOrEmailEnum.java new file mode 100644 index 00000000..d42e70a6 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/enums/AuthStrategyWhenNoUserWithPhoneOrEmailEnum.java @@ -0,0 +1,45 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.login.enums; + +import lombok.Getter; +import vip.xiaonuo.common.exception.CommonException; + +/** + * 手机号或邮箱无对应用户时策略 + * + * @author xuyuxiang + * @date 2021/10/11 14:02 + **/ +@Getter +public enum AuthStrategyWhenNoUserWithPhoneOrEmailEnum { + + /** 不允许登录 */ + NOT_ALLOW_LOGIN("NOT_ALLOW_LOGIN"), + + /** 自动创建用户 */ + AUTO_CREATE_USER("AUTO_CREATE_USER"); + + private final String value; + + AuthStrategyWhenNoUserWithPhoneOrEmailEnum(String value) { + this.value = value; + } + + public static void validate(String value) { + boolean flag = NOT_ALLOW_LOGIN.getValue().equals(value) || AUTO_CREATE_USER.getValue().equals(value); + if(!flag) { + throw new CommonException("不支持的手机号或邮箱无对应用户时策略型:{}", value); + } + } +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthEmailValidCodeLoginParam.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthEmailValidCodeLoginParam.java new file mode 100644 index 00000000..57017cfe --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthEmailValidCodeLoginParam.java @@ -0,0 +1,48 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.login.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 邮箱验证码登录参数 + * + * @author xuyuxiang + * @date 2022/7/7 16:46 + **/ +@Getter +@Setter +public class AuthEmailValidCodeLoginParam { + + /** 邮箱 */ + @Schema(description = "邮箱", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "邮箱不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "验证码不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "验证码请求号不能为空") + private String validCodeReqNo; + + /** 设备 */ + @Schema(description = "设备") + private String device; +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthGetEmailValidCodeParam.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthGetEmailValidCodeParam.java new file mode 100644 index 00000000..201838f6 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthGetEmailValidCodeParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.login.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 获取邮箱验证码参数 + * + * @author xuyuxiang + * @date 2022/8/25 13:45 + **/ +@Getter +@Setter +public class AuthGetEmailValidCodeParam { + + /** 邮箱 */ + @Schema(description = "邮箱", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "邮箱不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "验证码不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "验证码请求号不能为空") + private String validCodeReqNo; +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthRegisterParam.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthRegisterParam.java new file mode 100644 index 00000000..cf059a9d --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/param/AuthRegisterParam.java @@ -0,0 +1,47 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.login.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 注册参数 + * + * @author xuyuxiang + * @date 2022/7/7 16:46 + **/ +@Getter +@Setter +public class AuthRegisterParam { + + /** 账号 */ + @Schema(description = "账号", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "账号不能为空") + private String account; + + /** 密码 */ + @Schema(description = "密码", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "密码不能为空") + private String password; + + /** 验证码 */ + @Schema(description = "验证码") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + private String validCodeReqNo; +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/AuthService.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/AuthService.java index 4a6d7ac6..4bc69a6f 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/AuthService.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/AuthService.java @@ -14,9 +14,7 @@ package vip.xiaonuo.auth.modular.login.service; import vip.xiaonuo.auth.core.pojo.SaBaseClientLoginUser; import vip.xiaonuo.auth.core.pojo.SaBaseLoginUser; -import vip.xiaonuo.auth.modular.login.param.AuthAccountPasswordLoginParam; -import vip.xiaonuo.auth.modular.login.param.AuthGetPhoneValidCodeParam; -import vip.xiaonuo.auth.modular.login.param.AuthPhoneValidCodeLoginParam; +import vip.xiaonuo.auth.modular.login.param.*; import vip.xiaonuo.auth.modular.login.result.AuthPicValidCodeResult; /** @@ -36,13 +34,21 @@ public interface AuthService { AuthPicValidCodeResult getPicCaptcha(String type); /** - * 获取手机验证码 + * 获取手机登录验证码 * * @author xuyuxiang * @date 2021/12/28 14:46 **/ String getPhoneValidCode(AuthGetPhoneValidCodeParam authGetPhoneValidCodeParam, String type); + /** + * 获取邮箱登录验证码 + * + * @author xuyuxiang + * @date 2021/12/28 14:46 + **/ + String getEmailValidCode(AuthGetEmailValidCodeParam authGetEmailValidCodeParam, String type); + /** * 账号密码登录 * @@ -59,6 +65,14 @@ public interface AuthService { **/ String doLoginByPhone(AuthPhoneValidCodeLoginParam authPhoneValidCodeLoginParam, String type); + /** + * 邮箱验证码登录 + * + * @author xuyuxiang + * @date 2021/12/28 14:46 + **/ + String doLoginByEmail(AuthEmailValidCodeLoginParam authEmailValidCodeLoginParam, String type); + /** * 获取B端登录用户信息 * @@ -82,4 +96,12 @@ public interface AuthService { * @date 2022/7/9 14:44 */ String doLoginById(String userId, String device, String type); + + /** + * C端注册 + * + * @author xuyuxiang + * @date 2022/7/9 14:44 + */ + void register(AuthRegisterParam authRegisterParam, String type); } diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/impl/AuthServiceImpl.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/impl/AuthServiceImpl.java index 94da6e48..877d1982 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/impl/AuthServiceImpl.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/login/service/impl/AuthServiceImpl.java @@ -23,6 +23,7 @@ import cn.hutool.core.util.PhoneUtil; import cn.hutool.core.util.RandomUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.json.JSONObject; +import cn.hutool.json.JSONUtil; import com.baomidou.mybatisplus.core.toolkit.IdWorker; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; @@ -30,14 +31,15 @@ import vip.xiaonuo.auth.api.SaBaseLoginUserApi; import vip.xiaonuo.auth.core.enums.SaClientTypeEnum; import vip.xiaonuo.auth.core.pojo.SaBaseClientLoginUser; import vip.xiaonuo.auth.core.pojo.SaBaseLoginUser; +import vip.xiaonuo.auth.core.util.AuthEmailFormatUtl; import vip.xiaonuo.auth.core.util.StpClientLoginUserUtil; import vip.xiaonuo.auth.core.util.StpClientUtil; import vip.xiaonuo.auth.core.util.StpLoginUserUtil; import vip.xiaonuo.auth.modular.login.enums.AuthDeviceTypeEnum; import vip.xiaonuo.auth.modular.login.enums.AuthExceptionEnum; -import vip.xiaonuo.auth.modular.login.param.AuthAccountPasswordLoginParam; -import vip.xiaonuo.auth.modular.login.param.AuthGetPhoneValidCodeParam; -import vip.xiaonuo.auth.modular.login.param.AuthPhoneValidCodeLoginParam; +import vip.xiaonuo.auth.modular.login.enums.AuthPhoneOrEmailTypeEnum; +import vip.xiaonuo.auth.modular.login.enums.AuthStrategyWhenNoUserWithPhoneOrEmailEnum; +import vip.xiaonuo.auth.modular.login.param.*; import vip.xiaonuo.auth.modular.login.result.AuthPicValidCodeResult; import vip.xiaonuo.auth.modular.login.service.AuthService; import vip.xiaonuo.common.cache.CommonCacheOperator; @@ -45,7 +47,9 @@ import vip.xiaonuo.common.consts.CacheConstant; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.util.CommonCryptogramUtil; import vip.xiaonuo.common.util.CommonEmailUtil; +import vip.xiaonuo.common.util.CommonTimeFormatUtil; import vip.xiaonuo.dev.api.DevConfigApi; +import vip.xiaonuo.dev.api.DevEmailApi; import vip.xiaonuo.dev.api.DevSmsApi; import java.util.List; @@ -60,10 +64,82 @@ import java.util.stream.Collectors; @Service public class AuthServiceImpl implements AuthService { - private static final String SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_KEY = "SNOWY_SYS_DEFAULT_CAPTCHA_OPEN"; + /** B端验证码是否开启(适用图片验证码) */ + private static final String SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_B"; + /** C端验证码是否开启(适用图片验证码) */ + private static final String SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_C_KEY = "SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_C"; + + /** B端验证码失效时间(适用图片验证码和短信验证码,单位:分钟,默认5分钟有效) */ + private static final String SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_B_KEY = "SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_B"; + + /** C端验证码失效时间(适用图片验证码和短信验证码,单位:分钟,默认5分钟有效) */ + private static final String SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_C_KEY = "SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_C"; + + /** B端登录验证码短信消息模板编码 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_B"; + + /** C端登录验证码短信消息模板编码 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_C"; + + /** B端登录验证码邮件消息模板内容 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_B"; + + /** C端登录验证码邮件消息模板内容 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_C"; + + /** B端连续登录失败持续时间(即N分钟内连续登录失败,单位:分钟) */ + private static final String SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_B_KEY = "SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_B"; + + /** C端连续登录失败持续时间(即N分钟内连续登录失败,单位:分钟) */ + private static final String SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_C_KEY = "SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_C"; + + /** B端连续登录失败次数(即指定分钟内连续登录失败N次) */ + private static final String SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_B_KEY = "SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_B"; + + /** C端连续登录失败次数(即指定分钟内连续登录失败N次) */ + private static final String SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_C_KEY = "SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_C"; + + /** B端连续登录失败锁定时间(即指定分钟内连续登录失败指定次数,锁定N分钟,单位:分钟) */ + private static final String SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_B_KEY = "SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_B"; + + /** C端连续登录失败锁定时间(即指定分钟内连续登录失败指定次数,锁定N分钟,单位:分钟) */ + private static final String SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_C_KEY = "SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_C"; + + /** B端手机号登录是否开启 */ + private static final String SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_B"; + + /** C端手机号登录是否开启 */ + private static final String SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_C_KEY = "SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_C"; + + /** B端邮箱登录是否开启 */ + private static final String SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_B"; + + /** C端邮箱登录是否开启 */ + private static final String SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_C_KEY = "SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_C"; + + /** B端手机号无对应用户时策略 */ + private static final String SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_B_KEY = "SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_B"; + + /** C端手机号无对应用户时策略 */ + private static final String SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_C_KEY = "SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_C"; + + /** B端邮箱无对应用户时策略 */ + private static final String SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_B_KEY = "SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_B"; + + /** C端邮箱无对应用户时策略 */ + private static final String SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_C_KEY = "SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_C"; + + /** B端注册是否开启 */ + private static final String SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_B"; + + /** C端注册是否开启 */ + private static final String SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_C_KEY = "SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_C"; + + /** 验证码缓存前缀 */ private static final String AUTH_VALID_CODE_CACHE_KEY = "auth-validCode:"; + /** 失败次数缓存前缀 */ private static final String LOGIN_ERROR_TIMES_KEY_PREFIX = "login-error-times:"; @Resource(name = "loginUserApi") @@ -78,6 +154,9 @@ public class AuthServiceImpl implements AuthService { @Resource private DevSmsApi devSmsApi; + @Resource + private DevEmailApi devEmailApi; + @Resource private CommonCacheOperator commonCacheOperator; @@ -97,13 +176,17 @@ public class AuthServiceImpl implements AuthService { authPicValidCodeResult.setValidCodeBase64(validCodeBase64); // 将请求号返回前端 authPicValidCodeResult.setValidCodeReqNo(validCodeReqNo); - // 将请求号作为key,验证码的值作为value放到redis,用于校验,5分钟有效 - commonCacheOperator.put(AUTH_VALID_CODE_CACHE_KEY + validCodeReqNo, validCode, 5 * 60); + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(type); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(AUTH_VALID_CODE_CACHE_KEY + validCodeReqNo, validCode, validCodeExpiredDuration); return authPicValidCodeResult; } @Override public String getPhoneValidCode(AuthGetPhoneValidCodeParam authGetPhoneValidCodeParam, String type) { + // 校验是否允许手机号登录 + this.checkAllowPhoneLoginFlag(type); // 手机号 String phone = authGetPhoneValidCodeParam.getPhone(); // 验证码 @@ -111,26 +194,123 @@ public class AuthServiceImpl implements AuthService { // 验证码请求号 String validCodeReqNo = authGetPhoneValidCodeParam.getValidCodeReqNo(); // 校验参数 - validPhoneValidCodeParam(null, validCode, validCodeReqNo, type); + validPhoneOrEmailValidCodeParam(null, AuthPhoneOrEmailTypeEnum.PHONE.getValue(), validCode, validCodeReqNo, type); // 生成手机验证码的值,随机6为数字 String phoneValidCode = RandomUtil.randomNumbers(6); // 生成手机验证码的请求号 String phoneValidCodeReqNo = IdWorker.getIdStr(); - - // TODO 使用阿里云执行发送验证码,将验证码作为短信内容的参数变量放入, - // TODO 签名不传则使用系统默认配置的签名,支持传入多个参数,示例:{"name":"张三","number":"15038****76"} - //devSmsApi.sendSmsAliyun(phone, null, "验证码模板号", JSONUtil.toJsonStr(JSONUtil.createObj().set("validCode", phoneValidCode))); - - // TODO 使用腾讯云执行发送验证码,将验证码作为短信内容的参数变量放入, - // TODO sdkAppId和签名不传则使用系统默认配置的sdkAppId和签名,支持传入多个参数,逗号拼接,示例:"张三,15038****76,进行中" - devSmsApi.sendSmsTencent("sdkAppId", phone, "签名", "模板编码", phoneValidCode); - - // 将请求号作为key,验证码的值作为value放到redis,用于校验,5分钟有效 - commonCacheOperator.put(AUTH_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, 5 * 60); + // 登录验证码短信消息模板编码 + String smsTemplateCode; + if(SaClientTypeEnum.B.getValue().equals(type)) { + smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_B_KEY); + } else { + smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_C_KEY); + } + if(ObjectUtil.isEmpty(smsTemplateCode)){ + throw new CommonException("请联系管理员配置{}端登录验证码短信消息模板编码", type); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(type); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("validCode", phoneValidCode).set("validTime", validCodeExpiredDuration/60); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(AUTH_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, validCodeExpiredDuration); // 返回请求号 return phoneValidCodeReqNo; } + /** + * 校验是否允许手机号登录 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private void checkAllowPhoneLoginFlag(String type) { + // 是否允许手机号登录 + String allowPhoneLoginFlag; + if(SaClientTypeEnum.B.getValue().equals(type)) { + allowPhoneLoginFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_B_KEY); + } else { + allowPhoneLoginFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_C_KEY); + } + if(ObjectUtil.isNotEmpty(allowPhoneLoginFlag)) { + if(!Convert.toBool(allowPhoneLoginFlag)) { + throw new CommonException("管理员未开启手机号登录"); + } + } + } + + @Override + public String getEmailValidCode(AuthGetEmailValidCodeParam authGetEmailValidCodeParam, String type) { + // 校验是否允许邮箱登录 + this.checkAllowEmailLoginFlag(type); + // 邮箱 */ + String email = authGetEmailValidCodeParam.getEmail(); + // 验证码 + String validCode = authGetEmailValidCodeParam.getValidCode(); + // 验证码请求号 + String validCodeReqNo = authGetEmailValidCodeParam.getValidCodeReqNo(); + // 校验参数 + validPhoneOrEmailValidCodeParam(null, AuthPhoneOrEmailTypeEnum.EMAIL.getValue(), validCode, validCodeReqNo, type); + // 生成邮箱验证码的值,随机6为数字 + String emailValidCode = RandomUtil.randomNumbers(6); + // 生成邮箱验证码的请求号 + String emailValidCodeReqNo = IdWorker.getIdStr(); + // 登录验证码邮件消息模板内容 + String emailTemplateContent; + if(SaClientTypeEnum.B.getValue().equals(type)) { + emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_B_KEY); + } else { + emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_C_KEY); + } + if(ObjectUtil.isEmpty(emailTemplateContent)){ + throw new CommonException("请联系管理员配置{}端登录验证码邮件消息模板内容", type); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(type); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("validCode", emailValidCode).set("validTime", validCodeExpiredDuration/60); + // 获取格式化后的主题 + String subject = AuthEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = AuthEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(AUTH_VALID_CODE_CACHE_KEY + email + StrUtil.UNDERLINE + emailValidCodeReqNo, emailValidCode, validCodeExpiredDuration); + // 返回请求号 + return emailValidCodeReqNo; + } + + /** + * 校验是否允许邮箱登录 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private void checkAllowEmailLoginFlag(String type) { + // 是否允许邮箱登录 + String allowEmailLoginFlag; + if(SaClientTypeEnum.B.getValue().equals(type)) { + allowEmailLoginFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_B_KEY); + } else { + allowEmailLoginFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_C_KEY); + } + if(ObjectUtil.isNotEmpty(allowEmailLoginFlag)) { + if(!Convert.toBool(allowEmailLoginFlag)) { + throw new CommonException("管理员未开启邮箱登录"); + } + } + } + /** * 校验验证码方法 * @@ -141,8 +321,10 @@ public class AuthServiceImpl implements AuthService { // 依据请求号,取出缓存中的验证码进行校验 Object existValidCode; if(ObjectUtil.isEmpty(phoneOrEmail)) { + // 图形验证码 existValidCode = commonCacheOperator.get(AUTH_VALID_CODE_CACHE_KEY + validCodeReqNo); } else { + // 手机或者邮箱验证码 existValidCode = commonCacheOperator.get(AUTH_VALID_CODE_CACHE_KEY + phoneOrEmail + StrUtil.UNDERLINE + validCodeReqNo); } // 为空则直接验证码错误 @@ -151,8 +333,10 @@ public class AuthServiceImpl implements AuthService { } // 移除该验证码 if(ObjectUtil.isEmpty(phoneOrEmail)) { + // 图形验证码 commonCacheOperator.remove(AUTH_VALID_CODE_CACHE_KEY + validCodeReqNo); } else { + // 手机或者邮箱验证码 commonCacheOperator.remove(AUTH_VALID_CODE_CACHE_KEY + phoneOrEmail + StrUtil.UNDERLINE + validCodeReqNo); } // 不一致则直接验证码错误 @@ -167,28 +351,59 @@ public class AuthServiceImpl implements AuthService { * @author xuyuxiang * @date 2022/8/25 14:29 **/ - private void validPhoneValidCodeParam(String phoneOrEmail, String validCode, String validCodeReqNo, String type) { + private String validPhoneOrEmailValidCodeParam(String phoneOrEmail, String phoneOrEmailType, String validCode, + String validCodeReqNo, String type) { // 验证码正确则校验手机号格式 if(ObjectUtil.isEmpty(phoneOrEmail)) { - // 执行校验验证码 - validValidCode(null, validCode, validCodeReqNo); - } else { - if(!PhoneUtil.isMobile(phoneOrEmail) && !CommonEmailUtil.isEmail(phoneOrEmail)) { - throw new CommonException(AuthExceptionEnum.PHONE_FORMAT_ERROR.getValue()); - } - // 执行校验验证码 - validValidCode(phoneOrEmail, validCode, validCodeReqNo); - - // 根据手机号获取用户信息,判断用户是否存在,根据B端或C端判断 + // 根据手机号或者邮箱获取用户信息,判断用户是否存在,根据B端或C端判断 if(SaClientTypeEnum.B.getValue().equals(type)) { - if(ObjectUtil.isEmpty(loginUserApi.getUserByPhone(phoneOrEmail))) { - throw new CommonException(AuthExceptionEnum.PHONE_ERROR.getValue()); + if(phoneOrEmailType.equals(AuthPhoneOrEmailTypeEnum.PHONE.getValue())) { + SaBaseLoginUser saBaseLoginUser = loginUserApi.getUserByPhone(phoneOrEmail); + if(ObjectUtil.isEmpty(saBaseLoginUser)) { + // B端手机号无对应用户时策略 + return devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_B_KEY); + } + } else { + SaBaseLoginUser saBaseLoginUser = loginUserApi.getUserByEmail(phoneOrEmail); + if(ObjectUtil.isEmpty(saBaseLoginUser)) { + // B端邮箱无对应用户时策略 + return devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_B_KEY); + } } } else { - if(ObjectUtil.isEmpty(clientLoginUserApi.getClientUserByPhone(phoneOrEmail))) { - throw new CommonException(AuthExceptionEnum.PHONE_ERROR.getValue()); + if(phoneOrEmailType.equals(AuthPhoneOrEmailTypeEnum.PHONE.getValue())) { + SaBaseClientLoginUser saBaseClientLoginUser = clientLoginUserApi.getClientUserByPhone(phoneOrEmail); + if(ObjectUtil.isEmpty(saBaseClientLoginUser)) { + // C端手机号无对应用户时策略 + return devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_C_KEY); + } + } else { + SaBaseClientLoginUser saBaseClientLoginUser = clientLoginUserApi.getClientUserByEmail(phoneOrEmail); + if(ObjectUtil.isEmpty(saBaseClientLoginUser)) { + // BC端邮箱无对应用户时策略 + return devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_C_KEY); + } } } + // 执行校验图形验证码 + validValidCode(null, validCode, validCodeReqNo); + // 返回空值 + return null; + } else { + AuthPhoneOrEmailTypeEnum.validate(phoneOrEmailType); + if(phoneOrEmailType.equals(AuthPhoneOrEmailTypeEnum.PHONE.getValue())) { + if(!PhoneUtil.isMobile(phoneOrEmail)) { + throw new CommonException(AuthExceptionEnum.PHONE_FORMAT_ERROR.getValue()); + } + } else { + if(CommonEmailUtil.isNotEmail(phoneOrEmail)) { + throw new CommonException(AuthExceptionEnum.PHONE_FORMAT_ERROR.getValue()); + } + } + // 执行校验手机或者邮箱验证码 + validValidCode(phoneOrEmail, validCode, validCodeReqNo); + // 返回空值 + return null; } } @@ -209,7 +424,12 @@ public class AuthServiceImpl implements AuthService { AuthDeviceTypeEnum.validate(device); } // 校验验证码 - String defaultCaptchaOpen = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_KEY); + String defaultCaptchaOpen; + if(SaClientTypeEnum.B.getValue().equals(type)) { + defaultCaptchaOpen = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_B_KEY); + } else { + defaultCaptchaOpen = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_C_KEY); + } if(ObjectUtil.isNotEmpty(defaultCaptchaOpen)) { if(Convert.toBool(defaultCaptchaOpen)) { // 获取验证码 @@ -224,7 +444,7 @@ public class AuthServiceImpl implements AuthService { if(ObjectUtil.isEmpty(validCodeReqNo)) { throw new CommonException(AuthExceptionEnum.VALID_CODE_REQ_NO_EMPTY.getValue()); } - // 执行校验验证码 + // 执行校验图形验证码 validValidCode(null, validCode, validCodeReqNo); } } @@ -240,24 +460,24 @@ public class AuthServiceImpl implements AuthService { if(SaClientTypeEnum.B.getValue().equals(type)) { SaBaseLoginUser saBaseLoginUser = loginUserApi.getUserByAccount(account); if(ObjectUtil.isEmpty(saBaseLoginUser)) { + // 提示账号错误 throw new CommonException(AuthExceptionEnum.ACCOUNT_ERROR.getValue()); } if (!saBaseLoginUser.getPassword().equals(passwordHash)) { - // 记录登录次数 和 过期时间 - saveLoginTimes(account); - throw new CommonException(AuthExceptionEnum.PWD_ERROR.getValue()); + // 密码错误,处理剩余次数提示信息 + handleRemainingTimes(account, AuthExceptionEnum.PWD_ERROR.getValue(), type); } - // 删除redis 中的key - clearLoginErrorTimes(account); // 执行B端登录 return execLoginB(saBaseLoginUser, device); } else { SaBaseClientLoginUser saBaseClientLoginUser = clientLoginUserApi.getClientUserByAccount(account); if(ObjectUtil.isEmpty(saBaseClientLoginUser)) { + // 提示账号错误 throw new CommonException(AuthExceptionEnum.ACCOUNT_ERROR.getValue()); } if (!saBaseClientLoginUser.getPassword().equals(passwordHash)) { - throw new CommonException(AuthExceptionEnum.PWD_ERROR.getValue()); + // 密码错误,处理剩余次数提示信息 + handleRemainingTimes(account, AuthExceptionEnum.PWD_ERROR.getValue(), type); } // 执行C端登录 return execLoginC(saBaseClientLoginUser, device); @@ -266,10 +486,14 @@ public class AuthServiceImpl implements AuthService { @Override public String doLoginByPhone(AuthPhoneValidCodeLoginParam authPhoneValidCodeLoginParam, String type) { + // 校验是否允许手机号登录 + this.checkAllowPhoneLoginFlag(type); // 手机号 String phone = authPhoneValidCodeLoginParam.getPhone(); - // 校验参数 - validPhoneValidCodeParam(phone, authPhoneValidCodeLoginParam.getValidCode(), authPhoneValidCodeLoginParam.getValidCodeReqNo(), type); + // 校验参数,返回手机号无对应用户时的策略 + String strategyWhenNoUserWithPhoneOrEmail = validPhoneOrEmailValidCodeParam(phone, + AuthPhoneOrEmailTypeEnum.PHONE.getValue(), authPhoneValidCodeLoginParam.getValidCode(), + authPhoneValidCodeLoginParam.getValidCodeReqNo(), type); // 设备 String device = authPhoneValidCodeLoginParam.getDevice(); // 默认指定为PC,如在小程序跟移动端的情况下,自行指定即可 @@ -282,20 +506,115 @@ public class AuthServiceImpl implements AuthService { if(SaClientTypeEnum.B.getValue().equals(type)) { SaBaseLoginUser saBaseLoginUser = loginUserApi.getUserByPhone(phone); if(ObjectUtil.isEmpty(saBaseLoginUser)) { - throw new CommonException(AuthExceptionEnum.ACCOUNT_ERROR.getValue()); + // 判断手机号无对应用户时的策略,如果为空则直接抛出异常 + if(ObjectUtil.isEmpty(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("手机号码:{}不存在对应用户", phone); + } else { + // 如果不允许登录,则抛出异常 + if(AuthStrategyWhenNoUserWithPhoneOrEmailEnum.NOT_ALLOW_LOGIN.getValue().equals(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("手机号码:{}不存在对应用户", phone); + } else { + // 根据手机号自动创建B端用户 + saBaseLoginUser = loginUserApi.createUserWithPhone(phone); + } + } } // 执行B端登录 return execLoginB(saBaseLoginUser, device); } else { SaBaseClientLoginUser saBaseClientLoginUser = clientLoginUserApi.getClientUserByPhone(phone); if(ObjectUtil.isEmpty(saBaseClientLoginUser)) { - throw new CommonException(AuthExceptionEnum.ACCOUNT_ERROR.getValue()); + // 判断手机号无对应用户时的策略,如果为空则直接抛出异常 + if(ObjectUtil.isEmpty(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("手机号码:{}不存在对应用户", phone); + } else { + // 如果不允许登录,则抛出异常 + if(AuthStrategyWhenNoUserWithPhoneOrEmailEnum.NOT_ALLOW_LOGIN.getValue().equals(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("手机号码:{}不存在对应用户", phone); + } else { + // 根据手机号自动创建C端用户 + saBaseClientLoginUser = clientLoginUserApi.createClientUserWithPhone(phone); + } + } } // 执行C端登录 return execLoginC(saBaseClientLoginUser, device); } } + @Override + public String doLoginByEmail(AuthEmailValidCodeLoginParam authEmailValidCodeLoginParam, String type) { + // 校验是否允许邮箱登录 + this.checkAllowEmailLoginFlag(type); + // 邮箱 + String email = authEmailValidCodeLoginParam.getEmail(); + // 校验参数,返回邮箱无对应用户时的策略 + String strategyWhenNoUserWithPhoneOrEmail = validPhoneOrEmailValidCodeParam(email, + AuthPhoneOrEmailTypeEnum.EMAIL.getValue(), authEmailValidCodeLoginParam.getValidCode(), + authEmailValidCodeLoginParam.getValidCodeReqNo(), type); + // 设备 + String device = authEmailValidCodeLoginParam.getDevice(); + // 默认指定为PC,如在小程序跟移动端的情况下,自行指定即可 + if(ObjectUtil.isEmpty(device)) { + device = AuthDeviceTypeEnum.PC.getValue(); + } else { + AuthDeviceTypeEnum.validate(device); + } + // 根据邮箱获取用户信息,根据B端或C端判断 + if(SaClientTypeEnum.B.getValue().equals(type)) { + SaBaseLoginUser saBaseLoginUser = loginUserApi.getUserByEmail(email); + if(ObjectUtil.isEmpty(saBaseLoginUser)) { + // 判断邮箱无对应用户时的策略,如果为空则直接抛出异常 + if(ObjectUtil.isEmpty(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("邮箱:{}不存在对应用户", email); + } else { + // 如果不允许登录,则抛出异常 + if(AuthStrategyWhenNoUserWithPhoneOrEmailEnum.NOT_ALLOW_LOGIN.getValue().equals(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("邮箱:{}不存在对应用户", email); + } else { + // 根据邮箱自动创建B端用户 + saBaseLoginUser = loginUserApi.createUserWithEmail(email); + } + } + } + // 执行B端登录 + return execLoginB(saBaseLoginUser, device); + } else { + SaBaseClientLoginUser saBaseClientLoginUser = clientLoginUserApi.getClientUserByEmail(email); + if(ObjectUtil.isEmpty(saBaseClientLoginUser)) { + // 判断邮箱无对应用户时的策略,如果为空则直接抛出异常 + if(ObjectUtil.isEmpty(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("邮箱:{}不存在对应用户", email); + } else { + // 如果不允许登录,则抛出异常 + if(AuthStrategyWhenNoUserWithPhoneOrEmailEnum.NOT_ALLOW_LOGIN.getValue().equals(strategyWhenNoUserWithPhoneOrEmail)) { + throw new CommonException("邮箱:{}不存在对应用户", email); + } else { + // 根据邮箱自动创建C端用户 + saBaseClientLoginUser = clientLoginUserApi.createClientUserWithEmail(email); + } + } + } + // 执行C端登录 + return execLoginC(saBaseClientLoginUser, device); + } + } + + /** + * 处理剩余次数提示信息 + */ + private void handleRemainingTimes(String account, String errorMessage, String type) { + // 记录登录次数 和 过期时间 + int remainingTimes = saveLoginTimes(account, type); + if(remainingTimes == 0) { + // 此时已封禁,返回提示语 + isDisableTime(account); + } else { + // 提示错误 + throw new CommonException(errorMessage + ",您还可以尝试【" + remainingTimes + "】次"); + } + } + /** * 是否封禁状态 * 如果被封禁了,执行以下逻辑,返回前端还需等待的时间 @@ -304,33 +623,81 @@ public class AuthServiceImpl implements AuthService { // disableTime = -2表示未被封禁 long disableTime = StpUtil.getDisableTime(userAccount); if (disableTime > 0) { - if (disableTime > 60) { - throw new CommonException(userAccount + "账号已被封禁, 请再"+ disableTime/60+ "分钟后重新尝试登录!!"); - } - throw new CommonException(userAccount + "账号已被封禁, 请再"+ disableTime+ "秒后重新尝试登录!!"); + String formatTime = CommonTimeFormatUtil.formatSeconds(disableTime); + throw new CommonException("账号" + userAccount + "已被封禁, 请在"+ formatTime+ "后重新尝试登录!"); } } - // redis中保存登录错误次数 - private void saveLoginTimes(String userAccount){ - String loginErrorKey = LOGIN_ERROR_TIMES_KEY_PREFIX + userAccount; + /** + * redis中保存登录错误次数 + */ + private int saveLoginTimes(String userAccount, String type){ + // 获取连续登录失败持续时间 + String configContinuousLoginFailDuration; + // 获取连续登录失败次数 + String configContinuousLoginFailTimes; + // 获取连续登录失败锁定时间 + String configContinuousLoginFailLockDuration; + if(SaClientTypeEnum.B.getValue().equals(type)) { + configContinuousLoginFailDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_B_KEY); + configContinuousLoginFailTimes = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_B_KEY); + configContinuousLoginFailLockDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_B_KEY); + } else { + configContinuousLoginFailDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_C_KEY); + configContinuousLoginFailTimes = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_C_KEY); + configContinuousLoginFailLockDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_C_KEY); + } + // 连续登录失败持续时间默认5分钟 + long continuousLoginFailDuration = 5 * 60; + if(ObjectUtil.isNotEmpty(configContinuousLoginFailDuration)){ + // 配置了则使用配置的失效时间 + continuousLoginFailDuration = Convert.toLong(configContinuousLoginFailDuration) * 60; + } + + // 连续登录失败次数默认5次 + int continuousLoginFailTimes = 5; + if(ObjectUtil.isNotEmpty(configContinuousLoginFailTimes)){ + // 配置了则使用配置的失效时间 + continuousLoginFailTimes = Convert.toInt(configContinuousLoginFailTimes); + } + + // 连续登录失败锁定时间默认5分钟 + long continuousLoginFailLockDuration = 5 * 60; + if(ObjectUtil.isNotEmpty(configContinuousLoginFailLockDuration)){ + // 配置了则使用配置的失效时间 + continuousLoginFailLockDuration = Convert.toLong(configContinuousLoginFailLockDuration) * 60; + } + // 获取登录失败次数缓存键 + String loginErrorKey = LOGIN_ERROR_TIMES_KEY_PREFIX + ":" + userAccount; + // 获取登录失败次数缓存值 Integer number = (Integer) commonCacheOperator.get(loginErrorKey); if (number == null) { - // 如果redis中没有保存,代表失败第一次 - number = 2; - commonCacheOperator.put(loginErrorKey, number,5 * 60); - return; + // 如果redis中没有保存,代表失败第一次,如果配置的值为1次 + if(continuousLoginFailTimes == 1) { + // 直接进入isDisableTime方法,返回用户还需等待时间 + StpUtil.disable(userAccount, continuousLoginFailLockDuration); + // 删除redis 中的key + clearLoginErrorTimes(userAccount); + return 0; + } else { + // 否则失败次数为2 + number = 2; + commonCacheOperator.put(loginErrorKey, number, continuousLoginFailDuration); + return continuousLoginFailTimes - number + 1; + } + } else { + if (number < continuousLoginFailTimes) { + number++; + commonCacheOperator.put(loginErrorKey, number, continuousLoginFailDuration); + return continuousLoginFailTimes - number + 1; + } else { + // 第N次封禁账号,第N+1次进入isDisableTime方法,返回用户还需等待时间 + StpUtil.disable(userAccount, continuousLoginFailLockDuration); + // 删除redis 中的key + clearLoginErrorTimes(userAccount); + return 0; + } } - if (number < 5) { - number++; - commonCacheOperator.put(loginErrorKey, number,5 * 60); - return; - } - // 第五次封禁账号,第六次进入isDisableTime方法,返回用户还需等待时间 - StpUtil.disable(userAccount, 5 * 60); - // 删除redis 中的key - clearLoginErrorTimes(userAccount); - } /** @@ -338,7 +705,8 @@ public class AuthServiceImpl implements AuthService { * @param userAccount 账号 */ private void clearLoginErrorTimes(String userAccount) { - String loginErrorKey = LOGIN_ERROR_TIMES_KEY_PREFIX + userAccount; + // 获取登录失败次数缓存键 + String loginErrorKey = LOGIN_ERROR_TIMES_KEY_PREFIX + ":" + userAccount; // 删除redis中的key commonCacheOperator.remove(loginErrorKey); } @@ -466,11 +834,11 @@ public class AuthServiceImpl implements AuthService { // 填充B端用户信息并更新缓存 fillSaBaseLoginUserAndUpdateCache(saBaseLoginUser); // 去掉密码 - saBaseLoginUser.setPassword(null); + saBaseLoginUser.setPassword("******"); // 去掉权限码 - saBaseLoginUser.setPermissionCodeList(null); + saBaseLoginUser.setPermissionCodeList(CollectionUtil.newArrayList()); // 去掉数据范围 - saBaseLoginUser.setDataScopeList(null); + saBaseLoginUser.setDataScopeList(CollectionUtil.newArrayList()); // 返回 return saBaseLoginUser; } @@ -490,11 +858,11 @@ public class AuthServiceImpl implements AuthService { // 填充C端用户信息并更新缓存 fillSaBaseClientLoginUserAndUpdateCache(saBaseClientLoginUser); // 去掉密码 - saBaseClientLoginUser.setPassword(null); + saBaseClientLoginUser.setPassword("******"); // 去掉权限码 - saBaseClientLoginUser.setPermissionCodeList(null); + saBaseClientLoginUser.setPermissionCodeList(CollectionUtil.newArrayList()); // 去掉数据范围 - saBaseClientLoginUser.setDataScopeList(null); + saBaseClientLoginUser.setDataScopeList(CollectionUtil.newArrayList()); // 返回 return saBaseClientLoginUser; } @@ -518,4 +886,93 @@ public class AuthServiceImpl implements AuthService { return execLoginC(saBaseClientLoginUser, device); } } + + @Override + public void register(AuthRegisterParam authRegisterParam, String type) { + // 校验是否允许注册 + this.checkAllowRegisterFlag(type); + // 获取账号 + String account = authRegisterParam.getAccount(); + // 获取密码 + String password = authRegisterParam.getPassword(); + // 校验验证码 + String defaultCaptchaOpen; + if(SaClientTypeEnum.B.getValue().equals(type)) { + defaultCaptchaOpen = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_B_KEY); + } else { + defaultCaptchaOpen = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_C_KEY); + } + if(ObjectUtil.isNotEmpty(defaultCaptchaOpen)) { + if(Convert.toBool(defaultCaptchaOpen)) { + // 获取验证码 + String validCode = authRegisterParam.getValidCode(); + // 获取验证码请求号 + String validCodeReqNo = authRegisterParam.getValidCodeReqNo(); + // 开启验证码则必须传入验证码 + if(ObjectUtil.isEmpty(validCode)) { + throw new CommonException(AuthExceptionEnum.VALID_CODE_EMPTY.getValue()); + } + // 开启验证码则必须传入验证码请求号 + if(ObjectUtil.isEmpty(validCodeReqNo)) { + throw new CommonException(AuthExceptionEnum.VALID_CODE_REQ_NO_EMPTY.getValue()); + } + // 执行校验图形验证码 + validValidCode(null, validCode, validCodeReqNo); + } + } + // SM2解密前端传来的密码 + String passwordDecrypt = CommonCryptogramUtil.doSm2Decrypt(password); + // 根据账号获取用户信息,根据B端或C端判断 + if(SaClientTypeEnum.B.getValue().equals(type)) { + clientLoginUserApi.doRegister(account, passwordDecrypt); + } else { + loginUserApi.doRegister(account, passwordDecrypt); + } + } + + /** + * 校验是否开启注册 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private void checkAllowRegisterFlag(String type) { + // 是否允许注册 + String allowRegisterFlag; + if(SaClientTypeEnum.B.getValue().equals(type)) { + allowRegisterFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_B_KEY); + } else { + allowRegisterFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_C_KEY); + } + if(ObjectUtil.isNotEmpty(allowRegisterFlag)) { + if(!Convert.toBool(allowRegisterFlag)) { + throw new CommonException("管理员未开启注册"); + } + } + } + + /** + * 获取验证码失效时间(单位:秒) + * + * @author xuyuxiang + * @date 2025/3/21 20:25 + **/ + private long getValidCodeExpiredDuration(String type) { + // 默认5分钟 + int defaultExpiredTime = 5; + // 获取配置验证码失效时间(单位:分钟) + String configCaptchaExpiredDuration; + if(SaClientTypeEnum.B.getValue().equals(type)) { + configCaptchaExpiredDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_B_KEY); + } else { + configCaptchaExpiredDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_C_KEY); + } + // 判断是否为空 + if(ObjectUtil.isNotEmpty(configCaptchaExpiredDuration)){ + // 配置了则使用配置的失效时间 + defaultExpiredTime = Convert.toInt(configCaptchaExpiredDuration); + } + // 转为秒 + return defaultExpiredTime * 60L; + } } diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/monitor/controller/AuthSessionController.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/monitor/controller/AuthSessionController.java index 8374e3d1..fc189930 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/monitor/controller/AuthSessionController.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/monitor/controller/AuthSessionController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.auth.modular.monitor.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -32,6 +33,7 @@ import vip.xiaonuo.auth.modular.monitor.service.AuthSessionService; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +43,7 @@ import java.util.List; * @date 2022/6/24 15:20 **/ @Tag(name = "会话治理控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 3) @RestController @Validated public class AuthSessionController { @@ -54,6 +57,7 @@ public class AuthSessionController { * @author xuyuxiang * @date 2022/6/24 22:28 */ + @ApiOperationSupport(order = 1) @Operation(summary = "会话统计") @GetMapping("/auth/session/analysis") public CommonResult analysis() { @@ -66,6 +70,7 @@ public class AuthSessionController { * @author xuyuxiang * @date 2022/6/24 22:28 */ + @ApiOperationSupport(order = 2) @Operation(summary = "查询B端会话") @GetMapping("/auth/session/b/page") public CommonResult> pageForB(AuthSessionPageParam authSessionPageParam) { @@ -78,6 +83,7 @@ public class AuthSessionController { * @author xuyuxiang * @date 2022/6/24 22:28 */ + @ApiOperationSupport(order = 3) @Operation(summary = "查询C端会话") @GetMapping("/auth/session/c/page") public CommonResult> pageForC(AuthSessionPageParam authSessionPageParam) { @@ -90,11 +96,12 @@ public class AuthSessionController { * @author xuyuxiang * @date 2021/10/12 10:25 **/ + @ApiOperationSupport(order = 4) @Operation(summary = "强退B端会话") @CommonLog("强退B端会话") @PostMapping("/auth/session/b/exit") public CommonResult exitSessionForB(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List authExitSessionParamList) { + List authExitSessionParamList) { authSessionService.exitSessionForB(authExitSessionParamList); return CommonResult.ok(); } @@ -105,11 +112,12 @@ public class AuthSessionController { * @author xuyuxiang * @date 2021/10/12 10:25 **/ + @ApiOperationSupport(order = 5) @Operation(summary = "强退C端会话") @CommonLog("强退C端会话") @PostMapping("/auth/session/c/exit") public CommonResult exitSessionForC(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List authExitSessionParamList) { + List authExitSessionParamList) { authSessionService.exitSessionForC(authExitSessionParamList); return CommonResult.ok(); } @@ -120,6 +128,7 @@ public class AuthSessionController { * @author xuyuxiang * @date 2021/10/12 10:25 **/ + @ApiOperationSupport(order = 6) @Operation(summary = "强退B端token") @CommonLog("强退B端token") @PostMapping("/auth/token/b/exit") @@ -135,6 +144,7 @@ public class AuthSessionController { * @author xuyuxiang * @date 2021/10/12 10:25 **/ + @ApiOperationSupport(order = 7) @Operation(summary = "强退C端token") @CommonLog("强退C端token") @PostMapping("/auth/token/c/exit") diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/controller/AuthSsoController.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/controller/AuthSsoController.java new file mode 100644 index 00000000..b5e1e5e7 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/controller/AuthSsoController.java @@ -0,0 +1,58 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.sso.controller; + +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.annotation.Resource; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RestController; +import vip.xiaonuo.auth.core.enums.SaClientTypeEnum; +import vip.xiaonuo.auth.modular.sso.param.AuthSsoTicketLoginParam; +import vip.xiaonuo.auth.modular.sso.service.AuthSsoService; +import vip.xiaonuo.common.pojo.CommonResult; + +import javax.validation.Valid; + +/** + * 单点登录控制器 + * + * @author xuyuxiang + * @date 2022/8/30 9:20 + **/ +@Tag(name = "单点登录控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 4) +@RestController +@Validated +public class AuthSsoController { + + @Resource + private AuthSsoService authSsoService; + + /** + * 根据ticket执行单点登录 + * + * @author xuyuxiang + * @date 2021/10/15 13:12 + **/ + @ApiOperationSupport(order = 1) + @Operation(summary = "根据ticket执行单点登录") + @PostMapping("/auth/sso/doLogin") + public CommonResult doLogin(@RequestBody @Valid AuthSsoTicketLoginParam authAccountPasswordLoginParam) { + return CommonResult.data(authSsoService.doLogin(authAccountPasswordLoginParam, SaClientTypeEnum.B.getValue())); + } +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/param/AuthSsoTicketLoginParam.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/param/AuthSsoTicketLoginParam.java new file mode 100644 index 00000000..3b652bd4 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/param/AuthSsoTicketLoginParam.java @@ -0,0 +1,38 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.sso.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * ticket单点登录登录参数 + * + * @author xuyuxiang + * @date 2022/7/7 16:46 + **/ +@Getter +@Setter +public class AuthSsoTicketLoginParam { + + /** ticket */ + @Schema(description = "ticket", requiredMode = Schema.RequiredMode.REQUIRED) + @NotBlank(message = "ticket不能为空") + private String ticket; + + /** 设备 */ + @Schema(description = "设备") + private String device; +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/service/AuthSsoService.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/service/AuthSsoService.java new file mode 100644 index 00000000..96bbfb58 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/service/AuthSsoService.java @@ -0,0 +1,32 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.sso.service; + +import vip.xiaonuo.auth.modular.sso.param.AuthSsoTicketLoginParam; + +/** + * 单点登录Service接口 + * + * @author xuyuxiang + * @date 2022/8/30 9:20 + **/ +public interface AuthSsoService { + + /** + * 根据ticket执行单点登录 + * + * @author xuyuxiang + * @date 2022/8/30 9:36 + **/ + String doLogin(AuthSsoTicketLoginParam authAccountPasswordLoginParam, String value); +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/service/impl/AuthSsoServiceImpl.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/service/impl/AuthSsoServiceImpl.java new file mode 100644 index 00000000..bc8bf964 --- /dev/null +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/sso/service/impl/AuthSsoServiceImpl.java @@ -0,0 +1,37 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.auth.modular.sso.service.impl; + +import jakarta.annotation.Resource; +import org.springframework.stereotype.Service; +import vip.xiaonuo.auth.modular.login.service.AuthService; +import vip.xiaonuo.auth.modular.sso.param.AuthSsoTicketLoginParam; +import vip.xiaonuo.auth.modular.sso.service.AuthSsoService; + +/** + * 单点登录Service接口实现类 + * + * @author xuyuxiang + * @date 2022/8/30 9:21 + **/ +@Service +public class AuthSsoServiceImpl implements AuthSsoService { + + @Resource + private AuthService authService; + + @Override + public String doLogin(AuthSsoTicketLoginParam authAccountPasswordLoginParam, String device) { + return null; + } +} diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/controller/AuthThirdController.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/controller/AuthThirdController.java index 35dae66d..1a20569c 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/controller/AuthThirdController.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/controller/AuthThirdController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.auth.modular.third.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import me.zhyd.oauth.model.AuthCallback; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -29,6 +30,8 @@ import vip.xiaonuo.auth.modular.third.result.AuthThirdRenderResult; import vip.xiaonuo.auth.modular.third.service.AuthThirdService; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; + /** * 第三方登录控制器 * @@ -36,6 +39,7 @@ import vip.xiaonuo.common.pojo.CommonResult; * @date 2022/7/8 16:18 **/ @Tag(name = "三方登录控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 5) @RestController @Validated public class AuthThirdController { @@ -49,6 +53,7 @@ public class AuthThirdController { * @author xuyuxiang * @date 2022/7/8 16:19 **/ + @ApiOperationSupport(order = 1) @Operation(summary = "第三方登录页面渲染") @GetMapping("/auth/third/render") public CommonResult render(@Valid AuthThirdRenderParam authThirdRenderParam) { @@ -61,6 +66,7 @@ public class AuthThirdController { * @author xuyuxiang * @date 2022/7/8 16:42 **/ + @ApiOperationSupport(order = 2) @Operation(summary = "第三方登录授权回调") @GetMapping("/auth/third/callback") public CommonResult callback(@Valid AuthThirdCallbackParam authThirdCallbackParam, AuthCallback authCallback) { @@ -73,6 +79,7 @@ public class AuthThirdController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 3) @Operation(summary = "获取三方用户分页") @GetMapping("/auth/third/page") public CommonResult> page(AuthThirdUserPageParam authThirdUserPageParam) { diff --git a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/entity/AuthThirdUser.java b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/entity/AuthThirdUser.java index 6c997937..ae045266 100644 --- a/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/entity/AuthThirdUser.java +++ b/snowy-plugin/snowy-plugin-auth/src/main/java/vip/xiaonuo/auth/modular/third/entity/AuthThirdUser.java @@ -45,7 +45,7 @@ public class AuthThirdUser extends CommonEntity { /** 头像 */ @Schema(description = "头像") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String avatar; /** 姓名 */ @@ -54,12 +54,12 @@ public class AuthThirdUser extends CommonEntity { /** 昵称 */ @Schema(description = "昵称") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nickname; /** 性别 */ @Schema(description = "性别") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String gender; /** 分类 */ @@ -68,6 +68,6 @@ public class AuthThirdUser extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-biz/pom.xml b/snowy-plugin/snowy-plugin-biz/pom.xml index 1984227b..a3625a67 100644 --- a/snowy-plugin/snowy-plugin-biz/pom.xml +++ b/snowy-plugin/snowy-plugin-biz/pom.xml @@ -39,4 +39,4 @@ snowy-plugin-dev-api - \ No newline at end of file + diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/core/enums/BizBuildInEnum.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/core/enums/BizBuildInEnum.java index 363c1039..513f7af1 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/core/enums/BizBuildInEnum.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/core/enums/BizBuildInEnum.java @@ -23,7 +23,9 @@ import lombok.Getter; @Getter public enum BizBuildInEnum { - /** 超管用户账号 */ + /** + * 超管用户账号 + */ BUILD_IN_USER_ACCOUNT("superAdmin", "超管"); private final String value; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/controller/BizDictController.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/controller/BizDictController.java index 24e77628..8ca6a7de 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/controller/BizDictController.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/controller/BizDictController.java @@ -15,10 +15,11 @@ package vip.xiaonuo.biz.modular.dict.controller; import cn.dev33.satoken.annotation.SaCheckPermission; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; @@ -31,6 +32,7 @@ import vip.xiaonuo.biz.modular.dict.service.BizDictService; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; import java.util.List; /** @@ -40,6 +42,7 @@ import java.util.List; * @date 2022/6/21 14:58 **/ @Tag(name = "业务字典控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 4) @RestController @Validated public class BizDictController { @@ -53,6 +56,7 @@ public class BizDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取业务字典分页") @SaCheckPermission("/biz/dict/page") @GetMapping("/biz/dict/page") @@ -66,6 +70,7 @@ public class BizDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取业务字典树") @SaCheckPermission("/biz/dict/tree") @GetMapping("/biz/dict/tree") @@ -79,6 +84,7 @@ public class BizDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 3) @Operation(summary = "获取所有字典树") @GetMapping("/biz/dict/treeAll") public CommonResult>> treeAll() { @@ -91,6 +97,7 @@ public class BizDictController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "编辑业务字典") @CommonLog("编辑业务字典") @SaCheckPermission("/biz/dict/edit") diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/entity/BizDict.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/entity/BizDict.java index b4ce95ec..624b4c2e 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/entity/BizDict.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/entity/BizDict.java @@ -57,6 +57,6 @@ public class BizDict extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/param/BizDictEditParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/param/BizDictEditParam.java index 5e7c625d..4ebcc82f 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/param/BizDictEditParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/dict/param/BizDictEditParam.java @@ -29,17 +29,17 @@ import lombok.Setter; public class BizDictEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 字典文字 */ - @Schema(description = "字典文字", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "字典文字") @NotBlank(message = "dictLabel不能为空") private String dictLabel; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/controller/BizGroupController.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/controller/BizGroupController.java index 7ec8b1dd..a02e7584 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/controller/BizGroupController.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/controller/BizGroupController.java @@ -15,23 +15,23 @@ package vip.xiaonuo.biz.modular.group.controller; import cn.dev33.satoken.annotation.SaCheckPermission; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; -import io.swagger.v3.oas.annotations.tags.Tag; import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.annotation.Resource; +import jakarta.validation.Valid; +import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RestController; +import vip.xiaonuo.biz.modular.group.entity.BizGroup; import vip.xiaonuo.biz.modular.group.param.*; +import vip.xiaonuo.biz.modular.group.service.BizGroupService; import vip.xiaonuo.biz.modular.user.entity.BizUser; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; -import vip.xiaonuo.biz.modular.group.entity.BizGroup; -import vip.xiaonuo.biz.modular.group.service.BizGroupService; -import jakarta.annotation.Resource; -import jakarta.validation.Valid; -import jakarta.validation.constraints.NotEmpty; import java.util.List; /** diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/entity/BizGroup.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/entity/BizGroup.java index 6870cdae..db7bc1d6 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/entity/BizGroup.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/entity/BizGroup.java @@ -12,7 +12,8 @@ */ package vip.xiaonuo.biz.modular.group.entity; -import com.baomidou.mybatisplus.annotation.*; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; import io.swagger.v3.oas.annotations.media.Schema; import lombok.Getter; import lombok.Setter; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/mapper/mapping/BizGroupMapper.xml b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/mapper/mapping/BizGroupMapper.xml index 34b35a0a..1aa500dd 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/mapper/mapping/BizGroupMapper.xml +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/mapper/mapping/BizGroupMapper.xml @@ -2,4 +2,4 @@ - \ No newline at end of file + diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupAddParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupAddParam.java index 48093c10..e5dcb445 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupAddParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupAddParam.java @@ -13,10 +13,10 @@ package vip.xiaonuo.biz.modular.group.param; import io.swagger.v3.oas.annotations.media.Schema; -import lombok.Getter; -import lombok.Setter; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; +import lombok.Getter; +import lombok.Setter; /** * 用户组添加参数 diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupEditParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupEditParam.java index 9b14c16b..e812fc45 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupEditParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupEditParam.java @@ -13,10 +13,10 @@ package vip.xiaonuo.biz.modular.group.param; import io.swagger.v3.oas.annotations.media.Schema; -import lombok.Getter; -import lombok.Setter; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; +import lombok.Getter; +import lombok.Setter; /** * 用户组编辑参数 diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupIdParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupIdParam.java index 7df88659..ce18a546 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupIdParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/param/BizGroupIdParam.java @@ -13,11 +13,10 @@ package vip.xiaonuo.biz.modular.group.param; import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; import lombok.Getter; import lombok.Setter; -import jakarta.validation.constraints.NotBlank; - /** * 用户组Id参数 * diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/service/impl/BizGroupServiceImpl.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/service/impl/BizGroupServiceImpl.java index a178abe8..499e78c8 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/service/impl/BizGroupServiceImpl.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/group/service/impl/BizGroupServiceImpl.java @@ -21,7 +21,6 @@ import cn.hutool.core.lang.tree.TreeUtil; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.json.JSONUtil; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; @@ -29,17 +28,18 @@ import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import vip.xiaonuo.auth.core.util.StpLoginUserUtil; +import vip.xiaonuo.biz.modular.group.entity.BizGroup; +import vip.xiaonuo.biz.modular.group.mapper.BizGroupMapper; import vip.xiaonuo.biz.modular.group.param.*; +import vip.xiaonuo.biz.modular.group.service.BizGroupService; import vip.xiaonuo.biz.modular.org.entity.BizOrg; import vip.xiaonuo.biz.modular.org.service.BizOrgService; import vip.xiaonuo.biz.modular.user.entity.BizUser; +import vip.xiaonuo.biz.modular.user.enums.BizUserStatusEnum; import vip.xiaonuo.biz.modular.user.service.BizUserService; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.page.CommonPageRequest; -import vip.xiaonuo.biz.modular.group.entity.BizGroup; -import vip.xiaonuo.biz.modular.group.mapper.BizGroupMapper; -import vip.xiaonuo.biz.modular.group.service.BizGroupService; import vip.xiaonuo.sys.api.SysGroupApi; import java.util.List; @@ -143,32 +143,34 @@ public class BizGroupServiceImpl extends ServiceImpl i @Override public Page userSelector(BizGroupSelectorUserParam bizGroupSelectorUserParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + // 只查询状态为正常的 + queryWrapper.lambda().eq(BizUser::getUserStatus, BizUserStatusEnum.ENABLE.getValue()); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - lambdaQueryWrapper.in(BizUser::getOrgId, loginUserDataScope); + queryWrapper.lambda().in(BizUser::getOrgId, loginUserDataScope); } else { return new Page<>(); } // 只查询部分字段 - lambdaQueryWrapper.select(BizUser::getId, BizUser::getAvatar, BizUser::getOrgId, BizUser::getPositionId, BizUser::getAccount, + queryWrapper.lambda().select(BizUser::getId, BizUser::getAvatar, BizUser::getOrgId, BizUser::getPositionId, BizUser::getAccount, BizUser::getName, BizUser::getSortCode, BizUser::getGender, BizUser::getEntryDate); if (ObjectUtil.isNotEmpty(bizGroupSelectorUserParam.getOrgId())) { // 如果机构id不为空,则查询该机构及其子机构下的所有人 List childOrgIdList = CollStreamUtil.toList(bizOrgService.getChildListById(bizOrgService .getAllOrgList(), bizGroupSelectorUserParam.getOrgId(), true), BizOrg::getId); if (ObjectUtil.isNotEmpty(childOrgIdList)) { - lambdaQueryWrapper.in(BizUser::getOrgId, childOrgIdList); + queryWrapper.lambda().in(BizUser::getOrgId, childOrgIdList); } else { return new Page<>(); } } if(ObjectUtil.isNotEmpty(bizGroupSelectorUserParam.getSearchKey())) { - lambdaQueryWrapper.like(BizUser::getName, bizGroupSelectorUserParam.getSearchKey()); + queryWrapper.lambda().like(BizUser::getName, bizGroupSelectorUserParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(BizUser::getSortCode); - return bizUserService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(BizUser::getSortCode); + return bizUserService.page(CommonPageRequest.defaultPage(), queryWrapper); } @Override diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/controller/BizOrgController.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/controller/BizOrgController.java index 89cfbfaa..d87b7663 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/controller/BizOrgController.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/controller/BizOrgController.java @@ -15,10 +15,11 @@ package vip.xiaonuo.biz.modular.org.controller; import cn.dev33.satoken.annotation.SaCheckPermission; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -26,12 +27,14 @@ import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RestController; import vip.xiaonuo.biz.modular.org.entity.BizOrg; +import vip.xiaonuo.biz.modular.org.enums.BizOrgSourceFromTypeEnum; import vip.xiaonuo.biz.modular.org.param.*; import vip.xiaonuo.biz.modular.org.service.BizOrgService; import vip.xiaonuo.biz.modular.user.entity.BizUser; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +44,7 @@ import java.util.List; * @date 2022/4/24 19:55 */ @Tag(name = "机构控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 1) @RestController @Validated public class BizOrgController { @@ -54,6 +58,7 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取机构分页") @SaCheckPermission("/biz/org/page") @GetMapping("/biz/org/page") @@ -67,6 +72,7 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取机构树") @SaCheckPermission("/biz/org/tree") @GetMapping("/biz/org/tree") @@ -80,12 +86,13 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "添加机构") @CommonLog("添加机构") @SaCheckPermission("/biz/org/add") @PostMapping("/biz/org/add") public CommonResult add(@RequestBody @Valid BizOrgAddParam bizOrgAddParam) { - bizOrgService.add(bizOrgAddParam); + bizOrgService.add(bizOrgAddParam, BizOrgSourceFromTypeEnum.SYSTEM_ADD.getValue()); return CommonResult.ok(); } @@ -95,6 +102,7 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "编辑机构") @CommonLog("编辑机构") @SaCheckPermission("/biz/org/edit") @@ -110,6 +118,7 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "删除机构") @CommonLog("删除机构") @SaCheckPermission("/biz/org/delete") @@ -126,6 +135,7 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取机构详情") @SaCheckPermission("/biz/org/detail") @GetMapping("/biz/org/detail") @@ -141,6 +151,7 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取机构树选择器") @SaCheckPermission("/biz/org/orgTreeSelector") @GetMapping("/biz/org/orgTreeSelector") @@ -154,6 +165,7 @@ public class BizOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 8) @Operation(summary = "获取人员选择器") @SaCheckPermission("/biz/org/userSelector") @GetMapping("/biz/org/userSelector") diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/entity/BizOrg.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/entity/BizOrg.java index 727c19e9..1e3a0db7 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/entity/BizOrg.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/entity/BizOrg.java @@ -44,7 +44,7 @@ public class BizOrg extends CommonEntity { /** 主管id */ @Schema(description = "主管id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.SIMPLE, target = BizUser.class, fields = "name", alias = "director", ref = "directorName") private String directorId; @@ -66,6 +66,6 @@ public class BizOrg extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/entity/BizOrgExt.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/entity/BizOrgExt.java new file mode 100644 index 00000000..1700e092 --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/entity/BizOrgExt.java @@ -0,0 +1,46 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.org.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +/** + * 机构扩展实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("SYS_ORG_EXT") +public class BizOrgExt extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 机构id */ + @Schema(description = "机构id") + private String orgId; + + /** 来源类别 */ + @Schema(description = "来源类别") + private String sourceFromType; + +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/enums/BizOrgSourceFromTypeEnum.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/enums/BizOrgSourceFromTypeEnum.java new file mode 100644 index 00000000..679e1433 --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/enums/BizOrgSourceFromTypeEnum.java @@ -0,0 +1,37 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.org.enums; + +import lombok.Getter; + +/** + * 机构来源类型枚举 + * + * @author xuyuxiang + * @date 2022/4/21 19:56 + **/ +@Getter +public enum BizOrgSourceFromTypeEnum { + + /** 系统自建 */ + SYSTEM_ADD("SYSTEM_ADD"), + + /** 身份源 */ + ID_SOURCE("ID_SOURCE"); + + private final String value; + + BizOrgSourceFromTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/mapper/BizOrgExtMapper.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/mapper/BizOrgExtMapper.java new file mode 100644 index 00000000..7c85726b --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/mapper/BizOrgExtMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.org.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.biz.modular.org.entity.BizOrgExt; + +/** + * 机构扩展Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface BizOrgExtMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/mapper/mapping/BizOrgExtMapper.xml b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/mapper/mapping/BizOrgExtMapper.xml new file mode 100644 index 00000000..0289058a --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/mapper/mapping/BizOrgExtMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgAddParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgAddParam.java index 5377b0b1..baa7fbc5 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgAddParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgAddParam.java @@ -29,22 +29,22 @@ import lombok.Setter; public class BizOrgAddParam { /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgEditParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgEditParam.java index f967579c..bbce6e03 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgEditParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgEditParam.java @@ -29,27 +29,27 @@ import lombok.Setter; public class BizOrgEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgIdParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgIdParam.java index e6b68989..9c95cb61 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgIdParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/param/BizOrgIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class BizOrgIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/provider/BizOrgApiProvider.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/provider/BizOrgApiProvider.java index 2d4330db..58e89954 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/provider/BizOrgApiProvider.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/provider/BizOrgApiProvider.java @@ -21,6 +21,7 @@ import org.springframework.stereotype.Service; import vip.xiaonuo.biz.api.BizOrgApi; import vip.xiaonuo.biz.modular.org.param.BizOrgSelectorOrgListParam; import vip.xiaonuo.biz.modular.org.service.BizOrgService; + import java.util.List; /** diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/BizOrgExtService.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/BizOrgExtService.java new file mode 100644 index 00000000..ecc67ecd --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/BizOrgExtService.java @@ -0,0 +1,33 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.org.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.biz.modular.org.entity.BizOrgExt; + +/** + * 机构扩展Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface BizOrgExtService extends IService { + + /** + * 插入扩展信息 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void createExtInfo(String orgId, String sourceFromType); +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/BizOrgService.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/BizOrgService.java index 0f0209d4..69f621ab 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/BizOrgService.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/BizOrgService.java @@ -51,7 +51,7 @@ public interface BizOrgService extends IService { * @author xuyuxiang * @date 2022/4/24 20:48 */ - void add(BizOrgAddParam bizOrgAddParam); + void add(BizOrgAddParam bizOrgAddParam, String sourceFromType); /** * 编辑机构 diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/impl/BizOrgExtServiceImpl.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/impl/BizOrgExtServiceImpl.java new file mode 100644 index 00000000..d5478b72 --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/impl/BizOrgExtServiceImpl.java @@ -0,0 +1,37 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.org.service.impl; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import vip.xiaonuo.biz.modular.org.entity.BizOrgExt; +import vip.xiaonuo.biz.modular.org.mapper.BizOrgExtMapper; +import vip.xiaonuo.biz.modular.org.service.BizOrgExtService; + +/** + * 机构扩展Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class BizOrgExtServiceImpl extends ServiceImpl implements BizOrgExtService { + + @Override + public void createExtInfo(String orgId, String sourceFromType) { + BizOrgExt bizOrgExt = new BizOrgExt(); + bizOrgExt.setOrgId(orgId); + bizOrgExt.setSourceFromType(sourceFromType); + this.save(bizOrgExt); + } +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/impl/BizOrgServiceImpl.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/impl/BizOrgServiceImpl.java index b0d67242..b31c8826 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/impl/BizOrgServiceImpl.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/org/service/impl/BizOrgServiceImpl.java @@ -36,18 +36,19 @@ import vip.xiaonuo.biz.modular.org.entity.BizOrg; import vip.xiaonuo.biz.modular.org.enums.BizOrgCategoryEnum; import vip.xiaonuo.biz.modular.org.mapper.BizOrgMapper; import vip.xiaonuo.biz.modular.org.param.*; +import vip.xiaonuo.biz.modular.org.service.BizOrgExtService; import vip.xiaonuo.biz.modular.org.service.BizOrgService; import vip.xiaonuo.biz.modular.position.entity.BizPosition; import vip.xiaonuo.biz.modular.position.service.BizPositionService; import vip.xiaonuo.biz.modular.user.entity.BizUser; import vip.xiaonuo.biz.modular.user.service.BizUserService; -import vip.xiaonuo.common.cache.CommonCacheOperator; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.listener.CommonDataChangeEventCenter; import vip.xiaonuo.common.page.CommonPageRequest; import vip.xiaonuo.sys.api.SysRoleApi; +import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.Set; @@ -62,14 +63,12 @@ import java.util.stream.Collectors; @Service public class BizOrgServiceImpl extends ServiceImpl implements BizOrgService { - public static final String ORG_CACHE_ALL_KEY = "sys-org:all"; - - @Resource - private CommonCacheOperator commonCacheOperator; - @Resource private SysRoleApi sysRoleApi; + @Resource + private BizOrgExtService bizOrgExtService; + @Resource private BizPositionService bizPositionService; @@ -127,7 +126,7 @@ public class BizOrgServiceImpl extends ServiceImpl impleme @Transactional(rollbackFor = Exception.class) @Override - public void add(BizOrgAddParam bizOrgAddParam) { + public void add(BizOrgAddParam bizOrgAddParam, String sourceFromType) { BizOrgCategoryEnum.validate(bizOrgAddParam.getCategory()); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); @@ -147,8 +146,10 @@ public class BizOrgServiceImpl extends ServiceImpl impleme throw new CommonException("存在重复的同级机构,名称为:{}", bizOrg.getName()); } bizOrg.setCode(RandomUtil.randomString(10)); + // 保存机构 this.save(bizOrg); - + // 插入扩展信息 + bizOrgExtService.createExtInfo(bizOrg.getId(), sourceFromType); // 发布增加事件 CommonDataChangeEventCenter.doAddWithData(BizDataTypeEnum.ORG.getValue(), JSONUtil.createArray().put(bizOrg)); } @@ -182,8 +183,8 @@ public class BizOrgServiceImpl extends ServiceImpl impleme if(errorLevel) { throw new CommonException("不可选择上级机构:{}", this.getById(originDataList, bizOrg.getParentId()).getName()); } + // 更新机构 this.updateById(bizOrg); - // 发布更新事件 CommonDataChangeEventCenter.doUpdateWithData(BizDataTypeEnum.ORG.getValue(), JSONUtil.createArray().put(bizOrg)); } @@ -196,7 +197,7 @@ public class BizOrgServiceImpl extends ServiceImpl impleme // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - if(!loginUserDataScope.containsAll(orgIdList)) { + if(!new HashSet<>(loginUserDataScope).containsAll(orgIdList)) { throw new CommonException("您没有权限删除这些机构,机构id:{}", orgIdList); } } else { @@ -219,7 +220,7 @@ public class BizOrgServiceImpl extends ServiceImpl impleme List positionOrgIdList = CollectionUtil.newArrayList(); positionJsonList.forEach(positionJson -> JSONUtil.toList(JSONUtil.parseArray(positionJson), JSONObject.class) .forEach(jsonObject -> positionOrgIdList.add(jsonObject.getStr("orgId")))); - boolean hasPositionUser = CollectionUtil.intersectionDistinct(toDeleteOrgIdList, CollectionUtil.removeNull(positionOrgIdList)).size() > 0; + boolean hasPositionUser = !CollectionUtil.intersectionDistinct(toDeleteOrgIdList, CollectionUtil.removeNull(positionOrgIdList)).isEmpty(); if(hasPositionUser) { throw new CommonException("请先删除机构下的人员"); } @@ -263,28 +264,28 @@ public class BizOrgServiceImpl extends ServiceImpl impleme @Override public String getOrgIdByOrgFullNameWithCreate(String orgFullName) { - List cachedAllOrgList = this.getAllOrgList(); - List> treeList = TreeUtil.build(cachedAllOrgList.stream().map(bizOrg -> + List allOrgList = this.getAllOrgList(); + List> treeList = TreeUtil.build(allOrgList.stream().map(bizOrg -> new TreeNode<>(bizOrg.getId(), bizOrg.getParentId(), bizOrg.getName(), bizOrg.getSortCode())) .collect(Collectors.toList()), "0"); - return findOrgIdByOrgName("0", StrUtil.split(orgFullName, StrUtil.DASHED).iterator(), cachedAllOrgList, treeList); + return findOrgIdByOrgName("0", StrUtil.split(orgFullName, StrUtil.DASHED).iterator(), allOrgList, treeList); } - public String findOrgIdByOrgName(String parentId, Iterator iterator, List cachedAllOrgList, List> treeList) { + public String findOrgIdByOrgName(String parentId, Iterator iterator, List allOrgList, List> treeList) { String orgName = iterator.next(); if(ObjectUtil.isNotEmpty(treeList)) { List> findList = treeList.stream().filter(tree -> tree.getName().equals(orgName)).collect(Collectors.toList()); if(ObjectUtil.isNotEmpty(findList)) { if(iterator.hasNext()) { - return findOrgIdByOrgName(findList.get(0).getId(), iterator, cachedAllOrgList, findList.get(0).getChildren()); + return findOrgIdByOrgName(findList.get(0).getId(), iterator, allOrgList, findList.get(0).getChildren()); } else { return findList.get(0).getId(); } } } - String orgId = this.doCreateOrg(parentId, orgName, cachedAllOrgList); + String orgId = this.doCreateOrg(parentId, orgName, allOrgList); if(iterator.hasNext()) { - return findOrgIdByOrgName(orgId, iterator, cachedAllOrgList, CollectionUtil.newArrayList()); + return findOrgIdByOrgName(orgId, iterator, allOrgList, CollectionUtil.newArrayList()); } else { return orgId; } @@ -296,7 +297,7 @@ public class BizOrgServiceImpl extends ServiceImpl impleme * @author xuyuxiang * @date 2023/3/8 9:38 **/ - public String doCreateOrg(String parentId, String orgName, List cachedAllOrgList) { + public String doCreateOrg(String parentId, String orgName, List allOrgList) { //创建该机构 BizOrg bizOrg = new BizOrg(); bizOrg.setName(orgName); @@ -307,10 +308,6 @@ public class BizOrgServiceImpl extends ServiceImpl impleme this.save(bizOrg); // 发布增加事件 CommonDataChangeEventCenter.doAddWithData(BizDataTypeEnum.ORG.getValue(), JSONUtil.createArray().put(bizOrg)); - // 将该机构加入缓存 - cachedAllOrgList.add(bizOrg); - // 更新缓存 - commonCacheOperator.put(ORG_CACHE_ALL_KEY, cachedAllOrgList); return bizOrg.getId(); } @@ -342,55 +339,55 @@ public class BizOrgServiceImpl extends ServiceImpl impleme @Override public List orgListSelector(BizOrgSelectorOrgListParam bizOrgSelectorOrgListParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - lambdaQueryWrapper.in(BizOrg::getId, loginUserDataScope); + queryWrapper.lambda().in(BizOrg::getId, loginUserDataScope); } else { return CollectionUtil.newArrayList(); } // 查询部分字段 - lambdaQueryWrapper.select(BizOrg::getId, BizOrg::getParentId, BizOrg::getName, + queryWrapper.lambda().select(BizOrg::getId, BizOrg::getParentId, BizOrg::getName, BizOrg::getCategory, BizOrg::getSortCode); if(ObjectUtil.isNotEmpty(bizOrgSelectorOrgListParam.getParentId())) { - lambdaQueryWrapper.eq(BizOrg::getParentId, bizOrgSelectorOrgListParam.getParentId()); + queryWrapper.lambda().eq(BizOrg::getParentId, bizOrgSelectorOrgListParam.getParentId()); } if(ObjectUtil.isNotEmpty(bizOrgSelectorOrgListParam.getSearchKey())) { - lambdaQueryWrapper.like(BizOrg::getName, bizOrgSelectorOrgListParam.getSearchKey()); + queryWrapper.lambda().like(BizOrg::getName, bizOrgSelectorOrgListParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(BizOrg::getSortCode); - return this.list(lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(BizOrg::getSortCode); + return this.list(queryWrapper.lambda()); } @Override public Page userSelector(BizOrgSelectorUserParam bizOrgSelectorUserParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - lambdaQueryWrapper.in(BizUser::getOrgId, loginUserDataScope); + queryWrapper.lambda().in(BizUser::getOrgId, loginUserDataScope); } else { return new Page<>(); } // 只查询部分字段 - lambdaQueryWrapper.select(BizUser::getId, BizUser::getAvatar, BizUser::getOrgId, BizUser::getPositionId, BizUser::getAccount, + queryWrapper.lambda().select(BizUser::getId, BizUser::getAvatar, BizUser::getOrgId, BizUser::getPositionId, BizUser::getAccount, BizUser::getName, BizUser::getSortCode, BizUser::getGender, BizUser::getEntryDate); if (ObjectUtil.isNotEmpty(bizOrgSelectorUserParam.getOrgId())) { // 如果机构id不为空,则查询该机构及其子机构下的所有人 List childOrgIdList = CollStreamUtil.toList(this.getChildListById(this .getAllOrgList(), bizOrgSelectorUserParam.getOrgId(), true), BizOrg::getId); if (ObjectUtil.isNotEmpty(childOrgIdList)) { - lambdaQueryWrapper.in(BizUser::getOrgId, childOrgIdList); + queryWrapper.lambda().in(BizUser::getOrgId, childOrgIdList); } else { return new Page<>(); } } if(ObjectUtil.isNotEmpty(bizOrgSelectorUserParam.getSearchKey())) { - lambdaQueryWrapper.like(BizUser::getName, bizOrgSelectorUserParam.getSearchKey()); + queryWrapper.lambda().like(BizUser::getName, bizOrgSelectorUserParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(BizUser::getSortCode); - return bizUserService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(BizUser::getSortCode); + return bizUserService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } /* ====以下为各种递归方法==== */ diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/controller/BizPositionController.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/controller/BizPositionController.java index 6986cffa..4d7193f7 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/controller/BizPositionController.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/controller/BizPositionController.java @@ -15,10 +15,11 @@ package vip.xiaonuo.biz.modular.position.controller; import cn.dev33.satoken.annotation.SaCheckPermission; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -31,6 +32,7 @@ import vip.xiaonuo.biz.modular.position.service.BizPositionService; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; import java.util.List; /** @@ -40,6 +42,7 @@ import java.util.List; * @date 2022/4/25 20:40 */ @Tag(name = "岗位控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 2) @RestController @Validated public class BizPositionController { @@ -53,6 +56,7 @@ public class BizPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取岗位分页") @SaCheckPermission("/biz/position/page") @GetMapping("/biz/position/page") @@ -66,6 +70,7 @@ public class BizPositionController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加岗位") @CommonLog("添加岗位") @SaCheckPermission("/biz/position/add") @@ -81,6 +86,7 @@ public class BizPositionController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑岗位") @CommonLog("编辑岗位") @SaCheckPermission("/biz/position/edit") @@ -96,6 +102,7 @@ public class BizPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除岗位") @CommonLog("删除岗位") @SaCheckPermission("/biz/position/delete") @@ -112,6 +119,7 @@ public class BizPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取岗位详情") @SaCheckPermission("/biz/position/detail") @GetMapping("/biz/position/detail") @@ -127,6 +135,7 @@ public class BizPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取组织树选择器") @SaCheckPermission("/biz/position/orgTreeSelector") @GetMapping("/biz/position/orgTreeSelector") @@ -140,6 +149,7 @@ public class BizPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取岗位选择器") @SaCheckPermission("/biz/position/positionSelector") @GetMapping("/biz/position/positionSelector") diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/entity/BizPosition.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/entity/BizPosition.java index 8dad4e8a..7a4fb169 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/entity/BizPosition.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/entity/BizPosition.java @@ -57,6 +57,6 @@ public class BizPosition extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionAddParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionAddParam.java index 0ae96260..7bdb2515 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionAddParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionAddParam.java @@ -29,22 +29,22 @@ import lombok.Setter; public class BizPositionAddParam { /** 机构id */ - @Schema(description = "机构id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "机构id") @NotBlank(message = "orgId不能为空") private String orgId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionEditParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionEditParam.java index df5fb980..f3c24300 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionEditParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionEditParam.java @@ -29,27 +29,27 @@ import lombok.Setter; public class BizPositionEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 机构id */ - @Schema(description = "机构id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "机构id") @NotBlank(message = "orgId不能为空") private String orgId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionIdParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionIdParam.java index 5b7510ba..38e1dc20 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionIdParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/param/BizPositionIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class BizPositionIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/service/impl/BizPositionServiceImpl.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/service/impl/BizPositionServiceImpl.java index 2517bdc2..a1967ee8 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/service/impl/BizPositionServiceImpl.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/position/service/impl/BizPositionServiceImpl.java @@ -47,6 +47,7 @@ import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.listener.CommonDataChangeEventCenter; import vip.xiaonuo.common.page.CommonPageRequest; +import java.util.HashSet; import java.util.List; import java.util.Set; import java.util.stream.Collectors; @@ -71,7 +72,7 @@ public class BizPositionServiceImpl extends ServiceImpl queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 queryWrapper.lambda().select(BizPosition::getId, BizPosition::getOrgId, BizPosition::getName, - BizPosition::getCategory, BizPosition::getSortCode); + BizPosition::getCategory, BizPosition::getSortCode, BizPosition::getExtJson); if(ObjectUtil.isNotEmpty(bizPositionPageParam.getOrgId())) { queryWrapper.lambda().eq(BizPosition::getOrgId, bizPositionPageParam.getOrgId()); } @@ -162,7 +163,7 @@ public class BizPositionServiceImpl extends ServiceImpl loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - if(!loginUserDataScope.containsAll(positionOrgIdList)) { + if(!new HashSet<>(loginUserDataScope).containsAll(positionOrgIdList)) { throw new CommonException("您没有权限删除这些机构下的岗位,机构id:{}", positionOrgIdList); } } else { @@ -180,7 +181,7 @@ public class BizPositionServiceImpl extends ServiceImpl extPositionIdList = CollectionUtil.newArrayList(); positionJsonList.forEach(positionJson -> JSONUtil.toList(JSONUtil.parseArray(positionJson), JSONObject.class) .forEach(jsonObject -> extPositionIdList.add(jsonObject.getStr("positionId")))); - boolean hasPositionUser = CollectionUtil.intersectionDistinct(positionIdList, CollectionUtil.removeNull(extPositionIdList)).size() > 0; + boolean hasPositionUser = !CollectionUtil.intersectionDistinct(positionIdList, CollectionUtil.removeNull(extPositionIdList)).isEmpty(); if(hasPositionUser) { throw new CommonException("请先删除岗位下的用户"); } @@ -240,24 +241,24 @@ public class BizPositionServiceImpl extends ServiceImpl positionSelector(BizPositionSelectorPositionParam bizPositionSelectorPositionParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper(); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - lambdaQueryWrapper.in(BizPosition::getOrgId, loginUserDataScope); + queryWrapper.lambda().in(BizPosition::getOrgId, loginUserDataScope); } else { return new Page<>(); } // 查询部分字段 - lambdaQueryWrapper.select(BizPosition::getId, BizPosition::getOrgId, BizPosition::getName, + queryWrapper.lambda().select(BizPosition::getId, BizPosition::getOrgId, BizPosition::getName, BizPosition::getCategory, BizPosition::getSortCode); if(ObjectUtil.isNotEmpty(bizPositionSelectorPositionParam.getOrgId())) { - lambdaQueryWrapper.eq(BizPosition::getOrgId, bizPositionSelectorPositionParam.getOrgId()); + queryWrapper.lambda().eq(BizPosition::getOrgId, bizPositionSelectorPositionParam.getOrgId()); } if(ObjectUtil.isNotEmpty(bizPositionSelectorPositionParam.getSearchKey())) { - lambdaQueryWrapper.like(BizPosition::getName, bizPositionSelectorPositionParam.getSearchKey()); + queryWrapper.lambda().like(BizPosition::getName, bizPositionSelectorPositionParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(BizPosition::getSortCode); - return this.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(BizPosition::getSortCode); + return this.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/controller/BizUserController.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/controller/BizUserController.java index b0304a91..964b3142 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/controller/BizUserController.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/controller/BizUserController.java @@ -15,11 +15,12 @@ package vip.xiaonuo.biz.modular.user.controller; import cn.dev33.satoken.annotation.SaCheckPermission; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; import jakarta.servlet.http.HttpServletResponse; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.http.MediaType; import org.springframework.validation.annotation.Validated; @@ -30,12 +31,14 @@ import org.springframework.web.bind.annotation.RestController; import vip.xiaonuo.biz.modular.org.entity.BizOrg; import vip.xiaonuo.biz.modular.position.entity.BizPosition; import vip.xiaonuo.biz.modular.user.entity.BizUser; +import vip.xiaonuo.biz.modular.user.enums.BizUserSourceFromTypeEnum; import vip.xiaonuo.biz.modular.user.param.*; import vip.xiaonuo.biz.modular.user.result.BizUserRoleResult; import vip.xiaonuo.biz.modular.user.service.BizUserService; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; import java.io.IOException; import java.util.List; @@ -46,6 +49,7 @@ import java.util.List; * @date 2022/4/22 9:34 **/ @Tag(name = "人员控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 9) @RestController @Validated public class BizUserController { @@ -59,6 +63,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取人员分页") @SaCheckPermission("/biz/user/page") @GetMapping("/biz/user/page") @@ -72,12 +77,13 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加人员") @CommonLog("添加人员") @SaCheckPermission("/biz/user/add") @PostMapping("/biz/user/add") public CommonResult add(@RequestBody @Valid BizUserAddParam bizUserAddParam) { - bizUserService.add(bizUserAddParam); + bizUserService.add(bizUserAddParam, BizUserSourceFromTypeEnum.SYSTEM_ADD.getValue()); return CommonResult.ok(); } @@ -87,6 +93,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑人员") @CommonLog("编辑人员") @SaCheckPermission("/biz/user/edit") @@ -102,6 +109,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除人员") @CommonLog("删除人员") @SaCheckPermission("/biz/user/delete") @@ -118,6 +126,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取人员详情") @SaCheckPermission("/biz/user/detail") @GetMapping("/biz/user/detail") @@ -131,6 +140,7 @@ public class BizUserController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 6) @Operation(summary = "禁用人员") @CommonLog("禁用人员") @SaCheckPermission("/biz/user/disableUser") @@ -146,6 +156,7 @@ public class BizUserController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 7) @Operation(summary = "启用人员") @CommonLog("启用人员") @SaCheckPermission("/biz/user/enableUser") @@ -161,6 +172,7 @@ public class BizUserController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 8) @Operation(summary = "重置人员密码") @CommonLog("重置人员密码") @SaCheckPermission("/biz/user/resetPassword") @@ -176,6 +188,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 9) @Operation(summary = "获取人员拥有角色") @SaCheckPermission("/biz/user/ownRole") @GetMapping("/biz/user/ownRole") @@ -189,6 +202,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 10) @Operation(summary = "给人员授权角色") @CommonLog("给人员授权角色") @SaCheckPermission("/biz/user/grantRole") @@ -204,6 +218,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 11) @Operation(summary = "人员导出") @CommonLog("人员导出") @SaCheckPermission("/biz/user/export") @@ -218,6 +233,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 12) @Operation(summary = "导出人员个人信息") @CommonLog("导出人员个人信息") @SaCheckPermission("/biz/user/exportUserInfo") @@ -234,6 +250,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 13) @Operation(summary = "获取机构树选择器") @SaCheckPermission("/biz/user/orgTreeSelector") @GetMapping("/biz/user/orgTreeSelector") @@ -247,6 +264,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 14) @Operation(summary = "获取机构列表选择器") @SaCheckPermission("/biz/user/orgListSelector") @GetMapping("/biz/user/orgListSelector") @@ -260,6 +278,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 15) @Operation(summary = "获取岗位选择器") @SaCheckPermission("/biz/user/positionSelector") @GetMapping("/biz/user/positionSelector") @@ -273,6 +292,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 16) @Operation(summary = "获取角色选择器") @SaCheckPermission("/biz/user/roleSelector") @GetMapping("/biz/user/roleSelector") @@ -286,6 +306,7 @@ public class BizUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 17) @Operation(summary = "获取人员选择器") @SaCheckPermission("/biz/user/userSelector") @GetMapping("/biz/user/userSelector") diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/entity/BizUser.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/entity/BizUser.java index af95768a..f8b3e20c 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/entity/BizUser.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/entity/BizUser.java @@ -47,12 +47,12 @@ public class BizUser extends CommonEntity { /** 头像 */ @Schema(description = "头像") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String avatar; /** 签名 */ @Schema(description = "签名") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String signature; /** 账号 */ @@ -70,156 +70,154 @@ public class BizUser extends CommonEntity { /** 昵称 */ @Schema(description = "昵称") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nickname; /** 性别 */ @Schema(description = "性别") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.DICTIONARY, key = "GENDER") private String gender; /** 年龄 */ @Schema(description = "年龄") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String age; /** 出生日期 */ @Schema(description = "出生日期") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String birthday; /** 民族 */ @Schema(description = "民族") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nation; /** 籍贯 */ @Schema(description = "籍贯") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nativePlace; /** 家庭住址 */ @Schema(description = "家庭住址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String homeAddress; /** 通信地址 */ @Schema(description = "通信地址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String mailingAddress; /** 证件类型 */ @Schema(description = "证件类型") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String idCardType; /** 证件号码 */ @Schema(description = "证件号码") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String idCardNumber; /** 文化程度 */ @Schema(description = "文化程度") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String cultureLevel; /** 政治面貌 */ @Schema(description = "政治面貌") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String politicalOutlook; /** 毕业院校 */ @Schema(description = "毕业院校") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String college; /** 学历 */ @Schema(description = "学历") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String education; /** 学制 */ @Schema(description = "学制") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String eduLength; /** 学位 */ @Schema(description = "学位") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String degree; /** 手机 */ @Schema(description = "手机") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String phone; /** 邮箱 */ @Schema(description = "邮箱") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String email; /** 家庭电话 */ @Schema(description = "家庭电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String homeTel; /** 办公电话 */ @Schema(description = "办公电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String officeTel; /** 紧急联系人 */ @Schema(description = "紧急联系人") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String emergencyContact; /** 紧急联系人电话 */ @Schema(description = "紧急联系人电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String emergencyPhone; /** 紧急联系人地址 */ @Schema(description = "紧急联系人地址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String emergencyAddress; /** 员工编号 */ @Schema(description = "员工编号") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String empNo; /** 入职日期 */ @Schema(description = "入职日期") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String entryDate; /** 机构id */ @Schema(description = "机构id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) @Trans(type = TransType.SIMPLE, target = BizOrg.class, fields = "name", alias = "org", ref = "orgName") private String orgId; /** 岗位id */ @Schema(description = "岗位id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) @Trans(type = TransType.SIMPLE, target = BizPosition.class, fields = "name", alias = "position", ref = "positionName") private String positionId; /** 职级 */ @Schema(description = "职级") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String positionLevel; /** 主管id */ @Schema(description = "主管id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.SIMPLE, target = BizUser.class, fields = "name", alias = "director", ref = "directorName") private String directorId; /** 兼任信息 */ @Schema(description = "兼任信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String positionJson; /** 上次登录ip */ @@ -264,7 +262,7 @@ public class BizUser extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; @Schema(description = "机构名称") diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/entity/BizUserExt.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/entity/BizUserExt.java new file mode 100644 index 00000000..042d6915 --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/entity/BizUserExt.java @@ -0,0 +1,52 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.user.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +import java.util.Date; + +/** + * 用户扩展实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("SYS_USER_EXT") +public class BizUserExt extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 用户id */ + @Schema(description = "用户id") + private String userId; + + /** 来源类别 */ + @Schema(description = "来源类别") + private String sourceFromType; + + /** 密码修改日期 */ + @Schema(description = "密码修改日期") + private Date passwordUpdateTime; + +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/enums/BizUserSourceFromTypeEnum.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/enums/BizUserSourceFromTypeEnum.java new file mode 100644 index 00000000..8c124b58 --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/enums/BizUserSourceFromTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.user.enums; + +import lombok.Getter; + +/** + * 用户来源类型枚举 + * + * @author xuyuxiang + * @date 2022/4/21 19:56 + **/ +@Getter +public enum BizUserSourceFromTypeEnum { + + /** 系统自建 */ + SYSTEM_ADD("SYSTEM_ADD"), + + /** 用户注册 */ + SYSTEM_REGISTER("SYSTEM_REGISTER"), + + /** 身份源 */ + ID_SOURCE("ID_SOURCE"); + + private final String value; + + BizUserSourceFromTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/BizUserExtMapper.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/BizUserExtMapper.java new file mode 100644 index 00000000..85865b0c --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/BizUserExtMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.user.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.biz.modular.user.entity.BizUserExt; + +/** + * 用户扩展Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface BizUserExtMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/mapping/BizUserExtMapper.xml b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/mapping/BizUserExtMapper.xml new file mode 100644 index 00000000..cac6a04c --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/mapping/BizUserExtMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/mapping/BizUserMapper.xml b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/mapping/BizUserMapper.xml index 80c82584..6e97622f 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/mapping/BizUserMapper.xml +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/mapper/mapping/BizUserMapper.xml @@ -2,4 +2,4 @@ - \ No newline at end of file + diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserAddParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserAddParam.java index 1604bb61..ad0a732e 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserAddParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserAddParam.java @@ -28,22 +28,22 @@ import lombok.Setter; public class BizUserAddParam { /** 账号 */ - @Schema(description = "账号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "账号") @NotBlank(message = "account不能为空") private String account; /** 姓名 */ - @Schema(description = "姓名", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "姓名") @NotBlank(message = "name不能为空") private String name; /** 机构id */ - @Schema(description = "机构id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "机构id") @NotBlank(message = "orgId不能为空") private String orgId; /** 岗位id */ - @Schema(description = "岗位id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "岗位id") @NotBlank(message = "positionId不能为空") private String positionId; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserEditParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserEditParam.java index ee0e7c1a..ad4d938d 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserEditParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserEditParam.java @@ -28,27 +28,27 @@ import lombok.Setter; public class BizUserEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 账号 */ - @Schema(description = "账号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "账号") @NotBlank(message = "account不能为空") private String account; /** 姓名 */ - @Schema(description = "姓名", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "姓名") @NotBlank(message = "name不能为空") private String name; /** 机构id */ - @Schema(description = "机构id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "机构id") @NotBlank(message = "orgId不能为空") private String orgId; /** 岗位id */ - @Schema(description = "岗位id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "岗位id") @NotBlank(message = "positionId不能为空") private String positionId; diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserGrantRoleParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserGrantRoleParam.java index f7809015..d057dc6b 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserGrantRoleParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserGrantRoleParam.java @@ -31,12 +31,12 @@ import java.util.List; public class BizUserGrantRoleParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 角色id集合 */ - @Schema(description = "角色id集合", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "角色id集合") @NotNull(message = "roleIdList不能为空") private List roleIdList; } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserIdParam.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserIdParam.java index 21d7e643..4e7229ca 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserIdParam.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/param/BizUserIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class BizUserIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/BizUserExtService.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/BizUserExtService.java new file mode 100644 index 00000000..d5f2603d --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/BizUserExtService.java @@ -0,0 +1,41 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.user.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.biz.modular.user.entity.BizUserExt; + +/** + * 用户扩展Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface BizUserExtService extends IService { + + /** + * 更新用户最新修改密码时间 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void updatePasswordLastTime(String userId); + + /** + * 插入扩展信息 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void createExtInfo(String userId, String sourceFromType); +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/BizUserService.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/BizUserService.java index 1532f8c7..7c3906fc 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/BizUserService.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/BizUserService.java @@ -47,7 +47,7 @@ public interface BizUserService extends IService { * @author xuyuxiang * @date 2022/4/24 20:48 */ - void add(BizUserAddParam bizUserAddParam); + void add(BizUserAddParam bizUserAddParam, String sourceFromType); /** * 编辑人员 @@ -83,7 +83,7 @@ public interface BizUserService extends IService { /** * 禁用人员 - * + * * @author xuyuxiang * @date 2022/7/5 18:20 **/ diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/impl/BizUserExtServiceImpl.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/impl/BizUserExtServiceImpl.java new file mode 100644 index 00000000..e5dd949e --- /dev/null +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/impl/BizUserExtServiceImpl.java @@ -0,0 +1,55 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.biz.modular.user.service.impl; + +import cn.hutool.core.date.DateTime; +import cn.hutool.core.util.ObjectUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import vip.xiaonuo.biz.modular.user.entity.BizUserExt; +import vip.xiaonuo.biz.modular.user.mapper.BizUserExtMapper; +import vip.xiaonuo.biz.modular.user.service.BizUserExtService; + +/** + * 用户扩展Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class BizUserExtServiceImpl extends ServiceImpl implements BizUserExtService { + + @Override + public void updatePasswordLastTime(String userId) { + BizUserExt bizUserExt = this.getOne(new LambdaQueryWrapper().eq(BizUserExt::getUserId, userId)); + if(ObjectUtil.isEmpty(bizUserExt)){ + bizUserExt = new BizUserExt(); + bizUserExt.setUserId(userId); + bizUserExt.setPasswordUpdateTime(DateTime.now()); + this.save(bizUserExt); + } else { + bizUserExt.setPasswordUpdateTime(DateTime.now()); + this.updateById(bizUserExt); + } + } + + @Override + public void createExtInfo(String userId, String sourceFromType) { + BizUserExt bizUserExt = new BizUserExt(); + bizUserExt.setUserId(userId); + bizUserExt.setSourceFromType(sourceFromType); + bizUserExt.setPasswordUpdateTime(DateTime.now()); + this.save(bizUserExt); + } +} diff --git a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/impl/BizUserServiceImpl.java b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/impl/BizUserServiceImpl.java index 2ed9ff9b..16a8aa4f 100644 --- a/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/impl/BizUserServiceImpl.java +++ b/snowy-plugin/snowy-plugin-biz/src/main/java/vip/xiaonuo/biz/modular/user/service/impl/BizUserServiceImpl.java @@ -61,12 +61,14 @@ import vip.xiaonuo.biz.modular.org.service.BizOrgService; import vip.xiaonuo.biz.modular.position.entity.BizPosition; import vip.xiaonuo.biz.modular.position.service.BizPositionService; import vip.xiaonuo.biz.modular.user.entity.BizUser; +import vip.xiaonuo.biz.modular.user.entity.BizUserExt; import vip.xiaonuo.biz.modular.user.enums.BizRoleCategoryEnum; import vip.xiaonuo.biz.modular.user.enums.BizUserStatusEnum; import vip.xiaonuo.biz.modular.user.mapper.BizUserMapper; import vip.xiaonuo.biz.modular.user.param.*; import vip.xiaonuo.biz.modular.user.result.BizUserExportResult; import vip.xiaonuo.biz.modular.user.result.BizUserRoleResult; +import vip.xiaonuo.biz.modular.user.service.BizUserExtService; import vip.xiaonuo.biz.modular.user.service.BizUserService; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.excel.CommonExcelCustomMergeStrategy; @@ -74,7 +76,7 @@ import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.listener.CommonDataChangeEventCenter; import vip.xiaonuo.common.page.CommonPageRequest; import vip.xiaonuo.common.util.*; -import vip.xiaonuo.dev.api.DevConfigApi; +import vip.xiaonuo.sys.api.SysApi; import vip.xiaonuo.sys.api.SysRoleApi; import vip.xiaonuo.sys.api.SysUserApi; @@ -82,10 +84,7 @@ import java.io.BufferedOutputStream; import java.io.File; import java.io.IOException; import java.io.InputStream; -import java.util.Comparator; -import java.util.List; -import java.util.Map; -import java.util.Set; +import java.util.*; import java.util.stream.Collectors; /** @@ -97,13 +96,11 @@ import java.util.stream.Collectors; @Service public class BizUserServiceImpl extends ServiceImpl implements BizUserService { - private static final String SNOWY_SYS_DEFAULT_PASSWORD_KEY = "SNOWY_SYS_DEFAULT_PASSWORD"; - @Resource private TransService transService; @Resource - private DevConfigApi devConfigApi; + private SysApi sysApi; @Resource private SysUserApi sysUserApi; @@ -111,6 +108,9 @@ public class BizUserServiceImpl extends ServiceImpl impl @Resource private SysRoleApi sysRoleApi; + @Resource + private BizUserExtService bizUserExtService; + @Resource private BizOrgService bizOrgService; @@ -151,7 +151,7 @@ public class BizUserServiceImpl extends ServiceImpl impl @Transactional(rollbackFor = Exception.class) @Override - public void add(BizUserAddParam bizUserAddParam) { + public void add(BizUserAddParam bizUserAddParam, String sourceFromType) { checkParam(bizUserAddParam); BizUser bizUser = BeanUtil.toBean(bizUserAddParam, BizUser.class); if(ObjectUtil.isEmpty(bizUser.getAvatar())) { @@ -159,11 +159,13 @@ public class BizUserServiceImpl extends ServiceImpl impl bizUser.setAvatar(CommonAvatarUtil.generateImg(bizUser.getName())); } // 设置密码 - bizUser.setPassword(CommonCryptogramUtil.doHashValue(devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_KEY))); + bizUser.setPassword(CommonCryptogramUtil.doHashValue(sysApi.getDefaultPassword())); // 设置状态 bizUser.setUserStatus(BizUserStatusEnum.ENABLE.getValue()); + // 保存用户 this.save(bizUser); - + // 插入扩展信息 + bizUserExtService.createExtInfo(bizUser.getId(), sourceFromType); // 发布增加事件 CommonDataChangeEventCenter.doAddWithData(BizDataTypeEnum.USER.getValue(), JSONUtil.createArray().put(bizUser)); } @@ -192,7 +194,7 @@ public class BizUserServiceImpl extends ServiceImpl impl } } if(ObjectUtil.isNotEmpty(bizUserAddParam.getEmail())) { - if(!CommonEmailUtil.isEmail(bizUserAddParam.getEmail())) { + if(CommonEmailUtil.isNotEmail(bizUserAddParam.getEmail())) { throw new CommonException("邮箱:{}格式错误", bizUserAddParam.getEmail()); } if (this.count(new LambdaQueryWrapper() @@ -213,8 +215,8 @@ public class BizUserServiceImpl extends ServiceImpl impl throw new CommonException("不可修改系统内置超管人员账号"); } BeanUtil.copyProperties(bizUserEditParam, bizUser); + // 更新用户 this.updateById(bizUser); - // 发布更新事件 CommonDataChangeEventCenter.doUpdateWithData(BizDataTypeEnum.USER.getValue(), JSONUtil.createArray().put(bizUser)); } @@ -247,7 +249,7 @@ public class BizUserServiceImpl extends ServiceImpl impl } } if(ObjectUtil.isNotEmpty(bizUserEditParam.getEmail())) { - if(!CommonEmailUtil.isEmail(bizUserEditParam.getEmail())) { + if(CommonEmailUtil.isNotEmail(bizUserEditParam.getEmail())) { throw new CommonException("邮箱:{}格式错误", bizUserEditParam.getEmail()); } if (this.count(new LambdaQueryWrapper() @@ -273,7 +275,7 @@ public class BizUserServiceImpl extends ServiceImpl impl // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - if(!loginUserDataScope.containsAll(userOrgIdList)) { + if(!new HashSet<>(loginUserDataScope).containsAll(userOrgIdList)) { throw new CommonException("您没有权限删除这些机构下的人员,机构id:{}", CollectionUtil.subtract(userOrgIdList, loginUserDataScope)); } @@ -304,6 +306,9 @@ public class BizUserServiceImpl extends ServiceImpl impl // 执行删除 this.removeByIds(bizUserIdList); + // 删除扩展信息 + bizUserExtService.remove(new LambdaQueryWrapper().in(BizUserExt::getUserId, bizUserIdList)); + // 发布删除事件 CommonDataChangeEventCenter.doDeleteWithDataId(BizDataTypeEnum.USER.getValue(), bizUserIdList); } @@ -369,7 +374,7 @@ public class BizUserServiceImpl extends ServiceImpl impl } this.update(new LambdaUpdateWrapper().eq(BizUser::getId, bizUserIdParam.getId()).set(BizUser::getPassword, - CommonCryptogramUtil.doHashValue(devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_KEY)))); + CommonCryptogramUtil.doHashValue(sysApi.getDefaultPassword()))); } @Override @@ -563,7 +568,7 @@ public class BizUserServiceImpl extends ServiceImpl impl if(ObjectUtil.isNotEmpty(bizUser.getBirthday())) { try { // 年龄 - long age = cn.hutool.core.date.DateUtil.betweenYear(cn.hutool.core.date.DateUtil.parseDate(bizUser.getBirthday()), DateTime.now(), true); + long age = DateUtil.betweenYear(DateUtil.parseDate(bizUser.getBirthday()), DateTime.now(), true); if(age != 0) { map.put("age", age + "岁"); } @@ -633,48 +638,48 @@ public class BizUserServiceImpl extends ServiceImpl impl @Override public Page orgListSelector(BizUserSelectorOrgListParam bizUserSelectorOrgListParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - lambdaQueryWrapper.in(BizOrg::getId, loginUserDataScope); + queryWrapper.lambda().in(BizOrg::getId, loginUserDataScope); } else { return new Page<>(); } // 查询部分字段 - lambdaQueryWrapper.select(BizOrg::getId, BizOrg::getParentId, BizOrg::getName, + queryWrapper.lambda().select(BizOrg::getId, BizOrg::getParentId, BizOrg::getName, BizOrg::getCategory, BizOrg::getSortCode); if(ObjectUtil.isNotEmpty(bizUserSelectorOrgListParam.getParentId())) { - lambdaQueryWrapper.eq(BizOrg::getParentId, bizUserSelectorOrgListParam.getParentId()); + queryWrapper.lambda().eq(BizOrg::getParentId, bizUserSelectorOrgListParam.getParentId()); } if(ObjectUtil.isNotEmpty(bizUserSelectorOrgListParam.getSearchKey())) { - lambdaQueryWrapper.like(BizOrg::getName, bizUserSelectorOrgListParam.getSearchKey()); + queryWrapper.lambda().like(BizOrg::getName, bizUserSelectorOrgListParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(BizOrg::getSortCode); - return bizOrgService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(BizOrg::getSortCode); + return bizOrgService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } @Override public Page positionSelector(BizUserSelectorPositionParam bizUserSelectorPositionParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - lambdaQueryWrapper.in(BizPosition::getOrgId, loginUserDataScope); + queryWrapper.lambda().in(BizPosition::getOrgId, loginUserDataScope); } else { return new Page<>(); } // 查询部分字段 - lambdaQueryWrapper.select(BizPosition::getId, BizPosition::getOrgId, BizPosition::getName, + queryWrapper.lambda().select(BizPosition::getId, BizPosition::getOrgId, BizPosition::getName, BizPosition::getCategory, BizPosition::getSortCode); if(ObjectUtil.isNotEmpty(bizUserSelectorPositionParam.getOrgId())) { - lambdaQueryWrapper.eq(BizPosition::getOrgId, bizUserSelectorPositionParam.getOrgId()); + queryWrapper.lambda().eq(BizPosition::getOrgId, bizUserSelectorPositionParam.getOrgId()); } if(ObjectUtil.isNotEmpty(bizUserSelectorPositionParam.getSearchKey())) { - lambdaQueryWrapper.like(BizPosition::getName, bizUserSelectorPositionParam.getSearchKey()); + queryWrapper.lambda().like(BizPosition::getName, bizUserSelectorPositionParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(BizPosition::getSortCode); - return bizPositionService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(BizPosition::getSortCode); + return bizPositionService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } @SuppressWarnings("ALL") @@ -707,31 +712,31 @@ public class BizUserServiceImpl extends ServiceImpl impl @Override public Page userSelector(BizUserSelectorUserParam bizUserSelectorUserParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 校验数据范围 List loginUserDataScope = StpLoginUserUtil.getLoginUserDataScope(); if(ObjectUtil.isNotEmpty(loginUserDataScope)) { - lambdaQueryWrapper.in(BizUser::getOrgId, loginUserDataScope); + queryWrapper.lambda().in(BizUser::getOrgId, loginUserDataScope); } else { return new Page<>(); } // 只查询部分字段 - lambdaQueryWrapper.select(BizUser::getId, BizUser::getAvatar, BizUser::getOrgId, BizUser::getPositionId, BizUser::getAccount, + queryWrapper.lambda().select(BizUser::getId, BizUser::getAvatar, BizUser::getOrgId, BizUser::getPositionId, BizUser::getAccount, BizUser::getName, BizUser::getSortCode, BizUser::getGender, BizUser::getEntryDate); if (ObjectUtil.isNotEmpty(bizUserSelectorUserParam.getOrgId())) { // 如果机构id不为空,则查询该机构及其子机构下的所有人 List childOrgIdList = CollStreamUtil.toList(bizOrgService.getChildListById(bizOrgService .getAllOrgList(), bizUserSelectorUserParam.getOrgId(), true), BizOrg::getId); if (ObjectUtil.isNotEmpty(childOrgIdList)) { - lambdaQueryWrapper.in(BizUser::getOrgId, childOrgIdList); + queryWrapper.lambda().in(BizUser::getOrgId, childOrgIdList); } else { return new Page<>(); } } if(ObjectUtil.isNotEmpty(bizUserSelectorUserParam.getSearchKey())) { - lambdaQueryWrapper.like(BizUser::getName, bizUserSelectorUserParam.getSearchKey()); + queryWrapper.lambda().like(BizUser::getName, bizUserSelectorUserParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(BizUser::getSortCode); - return this.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(BizUser::getSortCode); + return this.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/enums/ClientPasswordComplexityEnum.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/enums/ClientPasswordComplexityEnum.java new file mode 100644 index 00000000..f7aca7e3 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/enums/ClientPasswordComplexityEnum.java @@ -0,0 +1,64 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.core.enums; + +import lombok.Getter; + +/** + * 密码复杂度类型枚举 + * + * @author xuyuxiang + * @date 2023/3/3 10:40 + **/ +@Getter +public enum ClientPasswordComplexityEnum { + + /** + * 无限制 + */ + REG0("REG0", "无限制"), + + /** + * 必须包含数字和字母 + */ + REG1("REG1", "必须包含数字和字母"), + + /** + * 必须包含数字和大写字母 + */ + REG2("REG2", "必须包含数字和大写字母"), + + /** + * 必须包含数字、大写字母、小写字母和特殊字符 + */ + REG3("REG3", "必须包含数字、大写字母、小写字母和特殊字符"), + + /** + * 至少包含数字、字母和特殊字符中的两种 + */ + REG4("REG4", "至少包含数字、字母和特殊字符中的两种"), + + /** + * 至少包含数字、大写字母、小写字母和特殊字符的三种 + */ + REG5("REG5", "至少包含数字、大写字母、小写字母和特殊字符的三种"); + + private final String value; + + private final String message; + + ClientPasswordComplexityEnum(String value, String message) { + this.value = value; + this.message = message; + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/timer/ClientUserPasswordExpiredNoticeTimerTaskRunner.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/timer/ClientUserPasswordExpiredNoticeTimerTaskRunner.java new file mode 100644 index 00000000..b8e3ca63 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/timer/ClientUserPasswordExpiredNoticeTimerTaskRunner.java @@ -0,0 +1,39 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.core.timer; + +import jakarta.annotation.Resource; +import lombok.extern.slf4j.Slf4j; +import org.springframework.stereotype.Component; +import vip.xiaonuo.client.modular.user.service.ClientUserService; +import vip.xiaonuo.common.timer.CommonTimerTaskRunner; + +/** + * 通知用户密码即将到期定时类 + * + * @author xuyuxiang + * @date 2022/8/5 15:52 + **/ +@Slf4j +@Component +public class ClientUserPasswordExpiredNoticeTimerTaskRunner implements CommonTimerTaskRunner { + + @Resource + private ClientUserService clientUserService; + + @Override + public void action(String extJson) { + // 通知用户密码即将到期 + clientUserService.noticeUserPasswordAboutToExpired(); + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/util/ClientEmailFormatUtl.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/util/ClientEmailFormatUtl.java new file mode 100644 index 00000000..5d641df8 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/util/ClientEmailFormatUtl.java @@ -0,0 +1,48 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.core.util; + +import cn.hutool.core.date.DateUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONObject; +import vip.xiaonuo.dev.api.DevConfigApi; + +/** + * 系统相关邮件格式化工具类 + * + * @author xuyuxiang + * @date 2025/3/21 19:07 + **/ +public class ClientEmailFormatUtl { + + /** 系统名称 */ + private static final String SNOWY_SYS_NAME_KEY = "SNOWY_SYS_NAME"; + + /** + * 格式化邮件内容 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static String format(String content, JSONObject paramMap) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 获取系统名称 + String sysName = devConfigApi.getValueByKey(SNOWY_SYS_NAME_KEY); + // 系统名称 + paramMap.set("sysName", sysName); + // 当前时间 + paramMap.set("sysNowTime", DateUtil.now()); + return StrUtil.format(content, paramMap); + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/util/ClientPasswordUtl.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/util/ClientPasswordUtl.java new file mode 100644 index 00000000..d8ccf707 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/core/util/ClientPasswordUtl.java @@ -0,0 +1,405 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.core.util; + +import cn.hutool.core.convert.Convert; +import cn.hutool.core.date.DateTime; +import cn.hutool.core.date.DateUnit; +import cn.hutool.core.date.DateUtil; +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.extra.pinyin.PinyinUtil; +import cn.hutool.extra.spring.SpringUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import vip.xiaonuo.client.core.enums.ClientPasswordComplexityEnum; +import vip.xiaonuo.client.modular.user.entity.ClientUser; +import vip.xiaonuo.client.modular.user.entity.ClientUserExt; +import vip.xiaonuo.client.modular.user.result.ClientLoginUser; +import vip.xiaonuo.client.modular.user.service.ClientUserExtService; +import vip.xiaonuo.client.modular.user.service.ClientUserPasswordService; +import vip.xiaonuo.client.modular.user.service.ClientUserService; +import vip.xiaonuo.common.exception.CommonException; +import vip.xiaonuo.common.util.CommonCryptogramUtil; +import vip.xiaonuo.dev.api.DevConfigApi; +import vip.xiaonuo.dev.api.DevWeakPasswordApi; + +import java.util.ArrayList; +import java.util.List; +import java.util.stream.Collectors; + +/** + * 系统密码工具类 + * + * @author xuyuxiang + * @date 2025/3/21 19:07 + **/ +public class ClientPasswordUtl { + + /** C端系统默认密码 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_FOR_C"; + + /** C端系统密码修改验证方式 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_C"; + + /** C端密码最小长度 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_C"; + + /** C端密码最大长度 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_C"; + + /** C端密码复杂度 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_C"; + + /** C端密码不能连续存在相同字符个数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_C_KEY= "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_C"; + + /** C端密码不能包含用户信息开关(开启后,密码中将不能包含账号、手机号、邮箱前缀和姓名拼音) */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_C"; + + /** C端密码不能使用历史密码开关 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_C"; + + /** C端密码不能使用历史密码范围个数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_C"; + + /** C端密码不能使用弱密码库中密码开关 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_C"; + + /** C端密码自定义额外弱密码库 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_C"; + + /** C端密码有效期天数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_C"; + + /** C端密码过期前提醒天数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_C_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_C"; + + /** + * 获取系统默认密码 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static String getDefaultPassword() { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + String defaultPassword = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_FOR_C_KEY); + if(ObjectUtil.isEmpty(defaultPassword)){ + throw new CommonException("请联系管理员配置系统默认密码"); + } + return defaultPassword; + } + + /** + * 校验修改密码验证方式 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validUpdatePasswordValidType(String type) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + String passwordUpdateValidType = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_C_KEY); + if(ObjectUtil.isEmpty(passwordUpdateValidType)){ + throw new CommonException("请联系管理员配置系统密码修改验证方式"); + } + if(!passwordUpdateValidType.equals(type)){ + throw new CommonException("系统配置不支持此方式修改密码"); + } + } + + /** + * 校验新密码 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(ClientLoginUser clientLoginUser, String newPassword) { + String userId = clientLoginUser.getId(); + String account = clientLoginUser.getAccount(); + String name = clientLoginUser.getName(); + String phone = clientLoginUser.getPhone(); + String email = clientLoginUser.getEmail(); + validNewPassword(userId, account, name, phone, email, newPassword); + } + + /** + * 校验新密码 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(ClientUser clientUser, String newPassword) { + String userId = clientUser.getId(); + String account = clientUser.getAccount(); + String name = clientUser.getName(); + String phone = clientUser.getPhone(); + String email = clientUser.getEmail(); + validNewPassword(userId, account, name, phone, email, newPassword); + } + + /** + * 校验新密码(基本要求) + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(String newPassword) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 密码最小长度 + String passwordMinLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordMinLength)){ + Integer minLengthInt = Convert.toInt(passwordMinLength); + if(newPassword.length() < minLengthInt){ + throw new CommonException("密码最小长度应为:{}", minLengthInt); + } + } + // 密码最大长度 + String passwordMaxLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordMaxLength)){ + Integer maxLengthInt = Convert.toInt(passwordMaxLength); + if(newPassword.length() > maxLengthInt){ + throw new CommonException("密码最大长度应为:{}", maxLengthInt); + } + } + // 密码复杂度 + String passwordComplexity = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordComplexity)){ + // 据复杂度配置判断 + boolean notLimit = ClientPasswordComplexityEnum.REG0.getValue().equals(passwordComplexity); + if(!notLimit){ + if(ClientPasswordComplexityEnum.REG1.getValue().equals(passwordComplexity)) { + String reg = "^(?=.*[0-9])(?=.*[a-zA-Z]).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", ClientPasswordComplexityEnum.REG1.getMessage()); + } + } + if(ClientPasswordComplexityEnum.REG2.getValue().equals(passwordComplexity)) { + String reg = "^(?=.*[0-9])(?=.*[A-Z]).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", ClientPasswordComplexityEnum.REG2.getMessage()); + } + } + if(ClientPasswordComplexityEnum.REG3.getValue().equals(passwordComplexity)) { + String reg = "^(?=.*[0-9])(?=.*[A-Z])(?=.*[a-z])(?=.*[^a-zA-Z0-9]).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", ClientPasswordComplexityEnum.REG3.getMessage()); + } + } + if(ClientPasswordComplexityEnum.REG4.getValue().equals(passwordComplexity)) { + String reg = "^(?:(?=.*[0-9])(?=.*[a-zA-Z])|(?=.*[0-9])(?=.*[^a-zA-Z0-9])|(?=.*[a-zA-Z])(?=.*[^a-zA-Z0-9])).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", ClientPasswordComplexityEnum.REG4.getMessage()); + } + } + if(ClientPasswordComplexityEnum.REG5.getValue().equals(passwordComplexity)) { + String reg = "^(?:(?=.*[0-9])(?=.*[A-Z])(?=.*[a-z])|(?=.*[0-9])(?=.*[A-Z])(?=.*[^a-zA-Z0-9])|(?=.*[0-9])(?=.*[a-z])(?=.*[^a-zA-Z0-9])|(?=.*[A-Z])(?=.*[a-z])(?=.*[^a-zA-Z0-9])).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", ClientPasswordComplexityEnum.REG5.getMessage()); + } + } + } + } + // 密码不能连续存在相同字符个数 + String passwordNotAllowContinuousSameCharacterLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowContinuousSameCharacterLength)){ + // 密码不能连续存在相同字符个数 + Integer passwordNotAllowContinuousSameCharacterLengthInt = Convert.toInt(passwordNotAllowContinuousSameCharacterLength); + boolean hasConsecutiveChars = hasConsecutiveChars(newPassword, passwordNotAllowContinuousSameCharacterLengthInt); + if(hasConsecutiveChars){ + throw new CommonException("密码不可存在" + passwordNotAllowContinuousSameCharacterLengthInt + "个连续相同的字符"); + } + } + + // 密码不能使用弱密码库中密码开关 + String passwordNotAllowUseWeakFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowUseWeakFlag)){ + if(Convert.toBool(passwordNotAllowUseWeakFlag)){ + // 不能包含内置弱密码和自定义额外弱密码库中密码 + DevWeakPasswordApi devWeakPasswordApi = SpringUtil.getBean(DevWeakPasswordApi.class); + List weakPasswordList = devWeakPasswordApi.weakPasswordList(); + // 获取自定义额外弱密码库 + String passwordDefineWeakDatabase = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordDefineWeakDatabase)){ + weakPasswordList.addAll(StrUtil.split(passwordDefineWeakDatabase, StrUtil.COMMA)); + } + if(weakPasswordList.contains(newPassword)){ + throw new CommonException("密码不可使用弱密码"); + } + } + } + } + + /** + * 校验新密码(含用户信息等) + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(String userId, String account, String name, String phone, String email, String newPassword) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 密码不能包含用户信息开关 + String passwordNotAllowContainsUserInfoFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowContainsUserInfoFlag)){ + if(Convert.toBool(passwordNotAllowContainsUserInfoFlag)){ + // 账号,不能包含 + if(ObjectUtil.isNotEmpty(account) && newPassword.contains(account)){ + throw new CommonException("密码不可包含账号"); + } + // 姓名,不能包含拼音 + if(ObjectUtil.isNotEmpty(name) && newPassword.contains(PinyinUtil.getPinyin(name, ""))){ + throw new CommonException("密码不可包含姓名拼音"); + } + // 手机号,不能包含 + if(ObjectUtil.isNotEmpty(phone) && newPassword.contains(phone)){ + throw new CommonException("密码不可包含手机号"); + } + // 邮箱,不能包含前缀,即@字符之前内容 + if(ObjectUtil.isNotEmpty(email) && newPassword.contains(StrUtil.split(email, StrUtil.AT).get(0))){ + throw new CommonException("密码不可包含邮箱前缀"); + } + } + } + // 密码不能使用历史密码开关 + String passwordNotAllowUseHistoryFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowUseHistoryFlag)){ + if(Convert.toBool(passwordNotAllowUseHistoryFlag)){ + // 密码不能使用历史密码范围个数 + String passwordNotAllowContainsHistoryCount = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowContainsHistoryCount)){ + Integer passwordNotAllowContainsHistoryCountInt = Convert.toInt(passwordNotAllowContainsHistoryCount); + // 取前N次密码,进行判断 + ClientUserPasswordService clientUserPasswordService = SpringUtil.getBean(ClientUserPasswordService.class); + List userPasswordHistoryLimit = clientUserPasswordService.getUserPasswordHistoryLimit(userId, passwordNotAllowContainsHistoryCountInt); + if(ObjectUtil.isNotEmpty(userPasswordHistoryLimit)){ + if(userPasswordHistoryLimit.contains(CommonCryptogramUtil.doHashValue(newPassword))) { + throw new CommonException("密码不可包含历史密码"); + } + } + } + } + } + } + + /** + * 判断当前用户密码是否过期 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static boolean isUserPasswordExpired(String userId) { + // 先判断是否系统默认密码 + ClientUserService clientUserService = SpringUtil.getBean(ClientUserService.class); + ClientUser clientUser = clientUserService.queryEntity(userId); + String defaultPassword = getDefaultPassword(); + // 若是,则认为已过期,需要修改 + if(CommonCryptogramUtil.doHashValue(defaultPassword).equals(clientUser.getPassword())){ + return true; + } + ClientUserExtService clientUserExtService = SpringUtil.getBean(ClientUserExtService.class); + // 获取用户扩展信息 + ClientUserExt clientUserExt = clientUserExtService.getOne(new LambdaQueryWrapper().eq(ClientUserExt::getUserId, userId)); + if(ObjectUtil.isEmpty(clientUserExt)){ + // 无扩展信息的,直接认定为密码过期 + return true; + } else { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 获取密码有效期天数 + String passwordExpiredDays = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_C_KEY); + if(ObjectUtil.isEmpty(passwordExpiredDays)){ + throw new CommonException("请联系管理员配置密码有效期天数"); + } else { + // 判断上次密码修改时间距今天的天数 + long betweenDays = DateUtil.between(clientUserExt.getPasswordUpdateTime(), DateTime.now(), DateUnit.DAY, false); + // 如果距离今天已经超过了系统配置的有效期,则认为已过期 + return betweenDays >= Convert.toInt(passwordExpiredDays); + } + } + } + + /** + * 获取今日需要提醒密码到期的用户集合 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static List thisDayPasswordExpiredNeedNoticeUserIdList() { + List resultUserList = new ArrayList<>(); + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 获取密码有效期天数 + String passwordExpiredDays = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_C_KEY); + // 密码有效期配置如果不为空 + if(ObjectUtil.isNotEmpty(passwordExpiredDays)){ + // 获取密码过期前提醒天数 + String passwordExpiredNoticeDays = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_C_KEY); + // 密码过期前提醒天数如果不为空 + if(ObjectUtil.isNotEmpty(passwordExpiredDays)){ + // 获取密码修改日期距离提醒日的天数N + int noticeOffsetDays = Convert.toInt(passwordExpiredDays) - Convert.toInt(passwordExpiredNoticeDays); + // 提前提醒天数必须大于过期天数 + if(noticeOffsetDays > 0) { + // 今日需要提醒的,即密码修改日期为N天前的 + DateTime passWordUpdateTime = DateUtil.offsetDay(DateTime.now(), -noticeOffsetDays); + // 获取当天的开始时间 + DateTime beginDay = DateUtil.beginOfDay(passWordUpdateTime); + // 获取当天的结束时间 + DateTime endDay = DateUtil.endOfDay(passWordUpdateTime); + // 获取ClientUserExtService + ClientUserExtService clientUserExtService = SpringUtil.getBean(ClientUserExtService.class); + // 查询密码修改日期在该范围内的(无修改密码历史(即扩展信息为空)的无需提醒,会强制修改密码) + List userIdList = clientUserExtService.list(new LambdaQueryWrapper() + .between(ClientUserExt::getPasswordUpdateTime,beginDay, endDay)).stream() + .map(ClientUserExt::getUserId).collect(Collectors.toList()); + if(ObjectUtil.isNotEmpty(userIdList)){ + // 获取ClientUserService + ClientUserService clientUserService = SpringUtil.getBean(ClientUserService.class); + resultUserList = clientUserService.listByIds(userIdList); + } + } + } + } + return resultUserList; + } + + /** + * 判断字符串是否存在N个连续相同的字符 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + private static boolean hasConsecutiveChars(String str, int n) { + // 处理无效输入:字符串为空、n非正、或字符串长度不足 + if (str == null || n <= 0 || str.length() < n) { + return false; + } + // 特殊情况:n=1时只要字符串非空即存在连续1个字符 + if (n == 1) { + return true; + } + + int maxStartIndex = str.length() - n; + for (int i = 0; i <= maxStartIndex; i++) { + char currentChar = str.charAt(i); + boolean isConsecutive = true; + // 检查后续n-1个字符是否相同 + for (int j = 1; j < n; j++) { + if (str.charAt(i + j) != currentChar) { + isConsecutive = false; + break; + } + } + if (isConsecutive) { + return true; + } + } + return false; + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/relation/service/impl/ClientRelationServiceImpl.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/relation/service/impl/ClientRelationServiceImpl.java index 70c57341..fd62bddd 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/relation/service/impl/ClientRelationServiceImpl.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/relation/service/impl/ClientRelationServiceImpl.java @@ -72,41 +72,49 @@ public class ClientRelationServiceImpl extends ServiceImpl targetIdList, String category) { this.saveRelationBatch(objectId, targetIdList, category, null, false); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithAppend(String objectId, List targetIdList, String category, List extJsonList) { this.saveRelationBatch(objectId, targetIdList, category, extJsonList, false); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationWithClear(String objectId, String targetId, String category) { this.saveRelation(objectId, targetId, category, null, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationWithClear(String objectId, String targetId, String category, String extJson) { this.saveRelation(objectId, targetId, category, extJson, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithClear(String objectId, List targetIdList, String category) { this.saveRelationBatch(objectId, targetIdList, category, null, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithClear(String objectId, List targetIdList, String category, List extJsonList) { this.saveRelationBatch(objectId, targetIdList, category, extJsonList, true); diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/controller/ClientUserCenterController.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/controller/ClientUserCenterController.java new file mode 100644 index 00000000..e589c18a --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/controller/ClientUserCenterController.java @@ -0,0 +1,363 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.controller; + +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.annotation.Resource; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; +import org.springframework.web.multipart.MultipartFile; +import vip.xiaonuo.client.modular.user.param.*; +import vip.xiaonuo.client.modular.user.result.ClientUserPicValidCodeResult; +import vip.xiaonuo.client.modular.user.service.ClientUserService; +import vip.xiaonuo.common.annotation.CommonLog; +import vip.xiaonuo.common.pojo.CommonResult; + +import javax.validation.Valid; + +/** + * C端用户个人控制器 + * + * @author xuyuxiang + * @date 2022/4/22 9:34 + **/ +@Tag(name = "C端用户个人控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 2) +@RestController +@Validated +public class ClientUserCenterController { + + @Resource + private ClientUserService clientUserService; + + /** + * 获取图片验证码 + * + * @author xuyuxiang + * @date 2022/7/8 9:26 + **/ + @ApiOperationSupport(order = 1) + @Operation(summary = "获取图片验证码") + @GetMapping("/client/userCenter/getPicCaptcha") + public CommonResult getPicCaptcha() { + return CommonResult.data(clientUserService.getPicCaptcha()); + } + + /** + * 找回密码获取手机验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 2) + @Operation(summary = "找回密码获取手机验证码") + @GetMapping("/client/userCenter/findPasswordGetPhoneValidCode") + public CommonResult findPasswordGetPhoneValidCode(@Valid ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam) { + return CommonResult.data(clientUserService.findPasswordGetPhoneValidCode(clientUserGetPhoneValidCodeParam)); + } + + /** + * 找回密码获取邮箱验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 3) + @Operation(summary = "找回密码获取邮箱验证码") + @GetMapping("/client/userCenter/findPasswordGetEmailValidCode") + public CommonResult findPasswordGetEmailValidCode(@Valid ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam) { + return CommonResult.data(clientUserService.findPasswordGetEmailValidCode(clientUserGetEmailValidCodeParam)); + } + + /** + * 通过手机号找回用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 4) + @Operation(summary = "通过手机号找回用户密码") + @CommonLog("通过手机号找回用户密码") + @PostMapping("/client/userCenter/findPasswordByPhone") + public CommonResult findPasswordByPhone(@RequestBody @Valid ClientUserFindPwdByPhoneParam clientUserFindPwdByPhoneParam) { + clientUserService.findPasswordByPhone(clientUserFindPwdByPhoneParam); + return CommonResult.ok(); + } + + /** + * 通过邮箱找回用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 5) + @Operation(summary = "通过邮箱找回用户密码") + @CommonLog("通过邮箱找回用户密码") + @PostMapping("/client/userCenter/findPasswordByEmail") + public CommonResult findPasswordByEmail(@RequestBody @Valid ClientUserFindPwdByEmailParam clientUserFindPwdByEmailParam) { + clientUserService.findPasswordByEmail(clientUserFindPwdByEmailParam); + return CommonResult.ok(); + } + + /** + * 修改密码获取手机验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 6) + @Operation(summary = "修改密码获取手机验证码") + @GetMapping("/client/userCenter/updatePasswordGetPhoneValidCode") + public CommonResult updatePasswordGetPhoneValidCode(@Valid ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam) { + return CommonResult.data(clientUserService.updatePasswordGetPhoneValidCode(clientUserGetPhoneValidCodeParam)); + } + + /** + * 修改密码获取邮箱验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 7) + @Operation(summary = "修改密码获取邮箱验证码") + @GetMapping("/client/userCenter/updatePasswordGetEmailValidCode") + public CommonResult updatePasswordGetEmailValidCode(@Valid ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam) { + return CommonResult.data(clientUserService.updatePasswordGetEmailValidCode(clientUserGetEmailValidCodeParam)); + } + + /** + * 通过验证旧密码修改用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 8) + @Operation(summary = "通过验证旧密码修改用户密码") + @CommonLog("通过验证旧密码修改用户密码") + @PostMapping("/client/userCenter/updatePasswordByOld") + public CommonResult updatePasswordByOld(@RequestBody @Valid ClientUserUpdatePwdByOldParam clientUserUpdatePwdByOldParam) { + clientUserService.updatePasswordByOld(clientUserUpdatePwdByOldParam); + return CommonResult.ok(); + } + + /** + * 通过验证手机号修改用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 9) + @Operation(summary = "通过验证手机号修改用户密码") + @CommonLog("通过验证手机号修改用户密码") + @PostMapping("/client/userCenter/updatePasswordByPhone") + public CommonResult updatePasswordByPhone(@RequestBody @Valid ClientUserUpdatePwdByPhoneParam clientUserUpdatePwdByPhoneParam) { + clientUserService.updatePasswordByPhone(clientUserUpdatePwdByPhoneParam); + return CommonResult.ok(); + } + + /** + * 通过验证邮箱修改用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 10) + @Operation(summary = "通过验证邮箱修改用户密码") + @CommonLog("通过验证邮箱修改用户密码") + @PostMapping("/client/userCenter/updatePasswordByEmail") + public CommonResult updatePasswordByEmail(@RequestBody @Valid ClientUserUpdatePwdByEmailParam clientUserUpdatePwdByEmailParam) { + clientUserService.updatePasswordByEmail(clientUserUpdatePwdByEmailParam); + return CommonResult.ok(); + } + + /** + * 绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 11) + @Operation(summary = "绑定手机号获取手机验证码") + @GetMapping("/client/userCenter/bindPhoneGetPhoneValidCode") + public CommonResult bindPhoneGetPhoneValidCode(@Valid ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam) { + return CommonResult.data(clientUserService.bindPhoneGetPhoneValidCode(clientUserGetPhoneValidCodeParam)); + } + + /** + * 修改绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 12) + @Operation(summary = "修改绑定手机号获取手机验证码") + @GetMapping("/client/userCenter/updateBindPhoneGetPhoneValidCode") + public CommonResult updateBindPhoneGetPhoneValidCode(@Valid ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam) { + return CommonResult.data(clientUserService.updateBindPhoneGetPhoneValidCode(clientUserGetPhoneValidCodeParam)); + } + + /** + * 绑定手机号 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 13) + @Operation(summary = "绑定手机号") + @CommonLog("绑定手机号") + @PostMapping("/client/userCenter/bindPhone") + public CommonResult bindPhone(@RequestBody @Valid ClientUserBindPhoneParam clientUserBindPhoneParam) { + clientUserService.bindPhone(clientUserBindPhoneParam); + return CommonResult.ok(); + } + + /** + * 绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 14) + @Operation(summary = "绑定邮箱获取邮箱验证码") + @GetMapping("/client/userCenter/bindEmailGetEmailValidCode") + public CommonResult bindEmailGetEmailValidCode(@Valid ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam) { + return CommonResult.data(clientUserService.bindEmailGetEmailValidCode(clientUserGetEmailValidCodeParam)); + } + + /** + * 修改绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 15) + @Operation(summary = "修改绑定邮箱获取邮箱验证码") + @GetMapping("/client/userCenter/updateBindEmailGetEmailValidCode") + public CommonResult updateBindEmailGetEmailValidCode(@Valid ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam) { + return CommonResult.data(clientUserService.updateBindEmailGetEmailValidCode(clientUserGetEmailValidCodeParam)); + } + + /** + * 绑定邮箱 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 16) + @Operation(summary = "绑定邮箱") + @CommonLog("绑定邮箱") + @PostMapping("/client/userCenter/bindEmail") + public CommonResult bindEmail(@RequestBody @Valid ClientUserBindEmailParam clientUserBindEmailParam) { + clientUserService.bindEmail(clientUserBindEmailParam); + return CommonResult.ok(); + } + + /** + * 修改用户头像 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 17) + @Operation(summary = "修改用户头像") + @CommonLog("修改用户头像") + @PostMapping("/client/userCenter/updateAvatar") + public CommonResult updateAvatar(@RequestPart("file") MultipartFile file) { + return CommonResult.data(clientUserService.updateAvatar(file)); + } + + /** + * 修改用户签名图片 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 18) + @Operation(summary = "修改用户签名图片") + @CommonLog("修改用户签名图片") + @PostMapping("/client/userCenter/updateSignature") + public CommonResult updateSignature(@RequestBody @Valid ClientUserSignatureParam clientUserSignatureParam) { + clientUserService.updateSignature(clientUserSignatureParam); + return CommonResult.ok(); + } + + /** + * 编辑个人信息 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 19) + @Operation(summary = "编辑个人信息") + @CommonLog("编辑个人信息") + @PostMapping("/client/userCenter/updateUserInfo") + public CommonResult updateUserInfo(@RequestBody @Valid ClientUserUpdateInfoParam clientUserUpdateInfoParam) { + clientUserService.updateUserInfo(clientUserUpdateInfoParam); + return CommonResult.ok(); + } + + /** + * 根据id获取头像 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 20) + @Operation(summary = "根据id获取头像") + @GetMapping("/client/userCenter/getAvatarById") + public CommonResult getAvatarById(@Valid ClientUserIdParam clientUserIdParam) { + return CommonResult.data(clientUserService.getAvatarById(clientUserIdParam)); + } + + /** + * 判断当前用户是否需要绑定手机号 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 21) + @Operation(summary = "判断当前用户是否需要绑定手机号") + @GetMapping("/client/userCenter/isUserNeedBindPhone") + public CommonResult isUserNeedBindPhone() { + return CommonResult.data(clientUserService.isUserNeedBindPhone()); + } + + /** + * 判断当前用户是否需要绑定邮箱 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 22) + @Operation(summary = "判断当前用户是否需要绑定邮箱") + @GetMapping("/client/userCenter/isUserNeedBindEmail") + public CommonResult isUserNeedBindEmail() { + return CommonResult.data(clientUserService.isUserNeedBindEmail()); + } + + /** + * 判断当前用户密码是否过期 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 21) + @Operation(summary = "判断当前用户密码是否过期") + @GetMapping("/client/userCenter/isUserPasswordExpired") + public CommonResult isUserPasswordExpired() { + return CommonResult.data(clientUserService.isUserPasswordExpired()); + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/controller/ClientUserController.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/controller/ClientUserController.java index d471048a..b36c8268 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/controller/ClientUserController.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/controller/ClientUserController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.client.modular.user.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -24,6 +25,7 @@ import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RestController; import vip.xiaonuo.client.modular.user.entity.ClientUser; +import vip.xiaonuo.client.modular.user.enums.ClientUserSourceFromTypeEnum; import vip.xiaonuo.client.modular.user.param.ClientUserAddParam; import vip.xiaonuo.client.modular.user.param.ClientUserEditParam; import vip.xiaonuo.client.modular.user.param.ClientUserIdParam; @@ -32,6 +34,7 @@ import vip.xiaonuo.client.modular.user.service.ClientUserService; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +44,7 @@ import java.util.List; * @date 2022/4/22 9:34 **/ @Tag(name = "C端用户控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 1) @RestController @Validated public class ClientUserController { @@ -49,39 +53,42 @@ public class ClientUserController { private ClientUserService clientUserService; /** - * 获取C端用户分页 + * 获取用户分页 * * @author xuyuxiang * @date 2022/4/24 20:00 */ - @Operation(summary = "获取C端用户分页") + @ApiOperationSupport(order = 1) + @Operation(summary = "获取用户分页") @GetMapping("/client/user/page") public CommonResult> page(ClientUserPageParam clientUserPageParam) { return CommonResult.data(clientUserService.page(clientUserPageParam)); } /** - * 添加C端用户 + * 添加用户 * * @author xuyuxiang * @date 2022/4/24 20:47 */ - @Operation(summary = "添加C端用户") - @CommonLog("添加C端用户") + @ApiOperationSupport(order = 2) + @Operation(summary = "添加用户") + @CommonLog("添加用户") @PostMapping("/client/user/add") public CommonResult add(@RequestBody @Valid ClientUserAddParam clientUserAddParam) { - clientUserService.add(clientUserAddParam); + clientUserService.add(clientUserAddParam, ClientUserSourceFromTypeEnum.SYSTEM_ADD.getValue()); return CommonResult.ok(); } /** - * 编辑C端用户 + * 编辑用户 * * @author xuyuxiang * @date 2022/4/24 20:47 */ - @Operation(summary = "编辑C端用户") - @CommonLog("编辑C端用户") + @ApiOperationSupport(order = 3) + @Operation(summary = "编辑用户") + @CommonLog("编辑用户") @PostMapping("/client/user/edit") public CommonResult edit(@RequestBody @Valid ClientUserEditParam clientUserEditParam) { clientUserService.edit(clientUserEditParam); @@ -89,27 +96,29 @@ public class ClientUserController { } /** - * 删除C端用户 + * 删除用户 * * @author xuyuxiang * @date 2022/4/24 20:00 */ - @Operation(summary = "删除C端用户") - @CommonLog("删除C端用户") + @ApiOperationSupport(order = 4) + @Operation(summary = "删除用户") + @CommonLog("删除用户") @PostMapping("/client/user/delete") public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List clientUserIdParamList) { + List clientUserIdParamList) { clientUserService.delete(clientUserIdParamList); return CommonResult.ok(); } /** - * 获取C端用户详情 + * 获取用户详情 * * @author xuyuxiang * @date 2022/4/24 20:00 */ - @Operation(summary = "获取C端用户详情") + @ApiOperationSupport(order = 5) + @Operation(summary = "获取用户详情") @GetMapping("/client/user/detail") public CommonResult detail(@Valid ClientUserIdParam clientUserIdParam) { return CommonResult.data(clientUserService.detail(clientUserIdParam)); diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUser.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUser.java index 36c9ef0d..4e8f3031 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUser.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUser.java @@ -43,12 +43,12 @@ public class ClientUser extends CommonEntity { /** 头像 */ @Schema(description = "头像,图片base64") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String avatar; /** 签名 */ @Schema(description = "签名,图片base64") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String signature; /** 账号 */ @@ -66,118 +66,118 @@ public class ClientUser extends CommonEntity { /** 昵称 */ @Schema(description = "昵称") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nickname; /** 性别 */ @Schema(description = "性别") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.DICTIONARY, key = "GENDER") private String gender; /** 年龄 */ @Schema(description = "年龄") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String age; /** 出生日期 */ @Schema(description = "出生日期") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String birthday; /** 民族 */ @Schema(description = "民族") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nation; /** 籍贯 */ @Schema(description = "籍贯") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nativePlace; /** 家庭住址 */ @Schema(description = "家庭住址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String homeAddress; /** 通信地址 */ @Schema(description = "通信地址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String mailingAddress; /** 证件类型 */ @Schema(description = "证件类型") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String idCardType; /** 证件号码 */ @Schema(description = "证件号码") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String idCardNumber; /** 文化程度 */ @Schema(description = "文化程度") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String cultureLevel; /** 政治面貌 */ @Schema(description = "政治面貌") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String politicalOutlook; /** 毕业院校 */ @Schema(description = "毕业院校") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String college; /** 学历 */ @Schema(description = "学历") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String education; /** 学制 */ @Schema(description = "学制") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String eduLength; /** 学位 */ @Schema(description = "学位") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String degree; /** 手机 */ @Schema(description = "手机") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String phone; /** 邮箱 */ @Schema(description = "邮箱") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String email; /** 家庭电话 */ @Schema(description = "家庭电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String homeTel; /** 办公电话 */ @Schema(description = "办公电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String officeTel; /** 紧急联系人 */ @Schema(description = "紧急联系人") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String emergencyContact; /** 紧急联系人电话 */ @Schema(description = "紧急联系人电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String emergencyPhone; /** 紧急联系人地址 */ @Schema(description = "紧急联系人地址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String emergencyAddress; /** 上次登录ip */ @@ -222,6 +222,6 @@ public class ClientUser extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUserExt.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUserExt.java new file mode 100644 index 00000000..4778facd --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUserExt.java @@ -0,0 +1,52 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +import java.util.Date; + +/** + * C端用户扩展实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("CLIENT_USER_EXT") +public class ClientUserExt extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 用户id */ + @Schema(description = "用户id") + private String userId; + + /** 来源类别 */ + @Schema(description = "来源类别") + private String sourceFromType; + + /** 密码修改日期 */ + @Schema(description = "密码修改日期") + private Date passwordUpdateTime; + +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUserPassword.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUserPassword.java new file mode 100644 index 00000000..550d2a54 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/entity/ClientUserPassword.java @@ -0,0 +1,46 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +/** + * C端用户密码实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("CLIENT_USER_PASSWORD") +public class ClientUserPassword extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 用户id */ + @Schema(description = "用户id") + private String userId; + + /** 密码 */ + @Schema(description = "密码") + private String password; + +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/enums/ClientUpdatePasswordValidTypeEnum.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/enums/ClientUpdatePasswordValidTypeEnum.java new file mode 100644 index 00000000..af468a1c --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/enums/ClientUpdatePasswordValidTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.enums; + +import lombok.Getter; + +/** + * 系统修改密码验证方式枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum ClientUpdatePasswordValidTypeEnum { + + /** 旧密码 */ + OLD("OLD"), + + /** 手机号 */ + PHONE("PHONE"), + + /** 邮箱 */ + EMAIL("EMAIL"); + + private final String value; + + ClientUpdatePasswordValidTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/enums/ClientUserSourceFromTypeEnum.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/enums/ClientUserSourceFromTypeEnum.java new file mode 100644 index 00000000..5f07de5d --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/enums/ClientUserSourceFromTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.enums; + +import lombok.Getter; + +/** + * 用户来源类型枚举 + * + * @author xuyuxiang + * @date 2022/4/21 19:56 + **/ +@Getter +public enum ClientUserSourceFromTypeEnum { + + /** 系统自建 */ + SYSTEM_ADD("SYSTEM_ADD"), + + /** 用户注册 */ + SYSTEM_REGISTER("SYSTEM_REGISTER"), + + /** 身份源 */ + ID_SOURCE("ID_SOURCE"); + + private final String value; + + ClientUserSourceFromTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/ClientUserExtMapper.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/ClientUserExtMapper.java new file mode 100644 index 00000000..46cd9371 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/ClientUserExtMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.client.modular.user.entity.ClientUserExt; + +/** + * C端用户扩展Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface ClientUserExtMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/ClientUserPasswordMapper.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/ClientUserPasswordMapper.java new file mode 100644 index 00000000..78862c8d --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/ClientUserPasswordMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.client.modular.user.entity.ClientUserPassword; + +/** + * C端用户密码Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface ClientUserPasswordMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/mapping/ClientUserExtMapper.xml b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/mapping/ClientUserExtMapper.xml new file mode 100644 index 00000000..d5550ce0 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/mapping/ClientUserExtMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/mapping/ClientUserPasswordMapper.xml b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/mapping/ClientUserPasswordMapper.xml new file mode 100644 index 00000000..1171b472 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/mapper/mapping/ClientUserPasswordMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserAddParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserAddParam.java index 5291c09f..99bfe877 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserAddParam.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserAddParam.java @@ -18,7 +18,7 @@ import lombok.Getter; import lombok.Setter; /** - * C端用户添加参数 + * 用户添加参数 * * @author xuyuxiang * @date 2022/4/21 16:13 @@ -37,6 +37,10 @@ public class ClientUserAddParam { @NotBlank(message = "name不能为空") private String name; + /** 密码 */ + @Schema(description = "密码") + private String password; + /** 头像 */ @Schema(description = "头像,图片base64") private String avatar; diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserBindEmailParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserBindEmailParam.java new file mode 100644 index 00000000..d950194d --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserBindEmailParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 绑定邮箱参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class ClientUserBindEmailParam { + + /** 邮箱 */ + @Schema(description = "邮箱") + @NotBlank(message = "email不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserBindPhoneParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserBindPhoneParam.java new file mode 100644 index 00000000..4b3cd4ea --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserBindPhoneParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 绑定手机号参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class ClientUserBindPhoneParam { + + /** 手机号 */ + @Schema(description = "手机号") + @NotBlank(message = "phone不能为空") + private String phone; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserEditParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserEditParam.java index d6ca910b..38a4dad7 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserEditParam.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserEditParam.java @@ -18,7 +18,7 @@ import lombok.Getter; import lombok.Setter; /** - * C端用户参数 + * 用户编辑参数 * * @author xuyuxiang * @date 2022/4/21 16:13 diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserFindPwdByEmailParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserFindPwdByEmailParam.java new file mode 100644 index 00000000..7ccbc7c7 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserFindPwdByEmailParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户找回密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class ClientUserFindPwdByEmailParam { + + /** 邮箱 */ + @Schema(description = "邮箱") + @NotBlank(message = "email不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserFindPwdByPhoneParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserFindPwdByPhoneParam.java new file mode 100644 index 00000000..64a6495c --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserFindPwdByPhoneParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户找回密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class ClientUserFindPwdByPhoneParam { + + /** 手机号 */ + @Schema(description = "手机号") + @NotBlank(message = "phone不能为空") + private String phone; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserGetEmailValidCodeParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserGetEmailValidCodeParam.java new file mode 100644 index 00000000..0e7db7e7 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserGetEmailValidCodeParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 获取邮箱验证码参数 + * + * @author xuyuxiang + * @date 2022/8/25 13:45 + **/ +@Getter +@Setter +public class ClientUserGetEmailValidCodeParam { + + /** 邮箱 */ + @Schema(description = "邮箱") + @NotBlank(message = "邮箱不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "验证码不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "验证码请求号不能为空") + private String validCodeReqNo; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserGetPhoneValidCodeParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserGetPhoneValidCodeParam.java new file mode 100644 index 00000000..f291fab5 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserGetPhoneValidCodeParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 获取手机验证码参数 + * + * @author xuyuxiang + * @date 2022/8/25 13:45 + **/ +@Getter +@Setter +public class ClientUserGetPhoneValidCodeParam { + + /** 手机号 */ + @Schema(description = "手机号") + @NotBlank(message = "手机号不能为空") + private String phone; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "验证码不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "验证码请求号不能为空") + private String validCodeReqNo; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserIdParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserIdParam.java index e98c7a89..fb28ad46 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserIdParam.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserIdParam.java @@ -18,7 +18,7 @@ import lombok.Getter; import lombok.Setter; /** - * C端用户Id参数 + * 用户Id参数 * * @author xuyuxiang * @date 2022/4/21 16:13 diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserPageParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserPageParam.java index 39bf173f..e04571c7 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserPageParam.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserPageParam.java @@ -17,7 +17,7 @@ import lombok.Getter; import lombok.Setter; /** - * C端用户参数 + * 用户查询参数 * * @author xuyuxiang * @date 2022/4/21 16:13 diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserSignatureParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserSignatureParam.java new file mode 100644 index 00000000..ff1d7f96 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserSignatureParam.java @@ -0,0 +1,34 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户修改签名图片参数 + * + * @author yubaoshan + * @date 2022/9/7 23:12 + **/ +@Getter +@Setter +public class ClientUserSignatureParam { + + /** 签名图片base64编码 */ + @Schema(description = "signature") + @NotBlank(message = "signature签名图片不能为空") + private String signature; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdateInfoParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdateInfoParam.java new file mode 100644 index 00000000..7b1b52bf --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdateInfoParam.java @@ -0,0 +1,51 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户编辑个人信息参数 + * + * @author xuyuxiang + * @date 2022/7/27 17:08 + **/ +@Getter +@Setter +public class ClientUserUpdateInfoParam { + + /** id */ + @Schema(description = "id") + @NotBlank(message = "id不能为空") + private String id; + + /** 姓名 */ + @Schema(description = "姓名") + @NotBlank(message = "name不能为空") + private String name; + + /** 昵称 */ + @Schema(description = "昵称") + private String nickname; + + /** 性别 */ + @Schema(description = "性别") + private String gender; + + /** 出生日期 */ + @Schema(description = "出生日期") + private String birthday; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByEmailParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByEmailParam.java new file mode 100644 index 00000000..adb206fe --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByEmailParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户通过验证邮箱修改密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class ClientUserUpdatePwdByEmailParam { + + /** 邮箱 */ + @Schema(description = "邮箱") + @NotBlank(message = "email不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByOldParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByOldParam.java new file mode 100644 index 00000000..2965efac --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByOldParam.java @@ -0,0 +1,39 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户通过验证旧密码修改密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class ClientUserUpdatePwdByOldParam { + + /** 旧密码 */ + @Schema(description = "旧密码") + @NotBlank(message = "password不能为空") + private String password; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByPhoneParam.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByPhoneParam.java new file mode 100644 index 00000000..75321853 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/param/ClientUserUpdatePwdByPhoneParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户通过验证手机号修改密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class ClientUserUpdatePwdByPhoneParam { + + /** 手机号 */ + @Schema(description = "手机号") + @NotBlank(message = "phone不能为空") + private String phone; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/provider/ClientLoginUserApiProvider.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/provider/ClientLoginUserApiProvider.java index a71e838a..4b99f058 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/provider/ClientLoginUserApiProvider.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/provider/ClientLoginUserApiProvider.java @@ -12,6 +12,7 @@ */ package vip.xiaonuo.client.modular.user.provider; +import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.collection.CollectionUtil; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; @@ -20,6 +21,7 @@ import org.springframework.stereotype.Service; import vip.xiaonuo.auth.api.SaBaseLoginUserApi; import vip.xiaonuo.auth.core.pojo.SaBaseClientLoginUser; import vip.xiaonuo.auth.core.pojo.SaBaseLoginUser; +import vip.xiaonuo.client.modular.user.entity.ClientUser; import vip.xiaonuo.client.modular.user.result.ClientLoginUser; import vip.xiaonuo.client.modular.user.service.ClientUserService; @@ -93,6 +95,11 @@ public class ClientLoginUserApiProvider implements SaBaseLoginUserApi { return null; } + @Override + public SaBaseLoginUser getUserByEmail(String email) { + return null; + } + /** * 根据手机号获取C端用户信息,查不到则返回null * @@ -104,6 +111,11 @@ public class ClientLoginUserApiProvider implements SaBaseLoginUserApi { return clientUserService.getUserByPhone(phone); } + @Override + public SaBaseClientLoginUser getClientUserByEmail(String email) { + return clientUserService.getUserByEmail(email); + } + /** * 根据用户id获取用户集合 * @@ -173,4 +185,31 @@ public class ClientLoginUserApiProvider implements SaBaseLoginUserApi { public void updateUserLoginInfo(String userId, String device) { clientUserService.updateUserLoginInfo(userId, device); } -} \ No newline at end of file + + @Override + public SaBaseLoginUser createUserWithPhone(String phone) { + return null; + } + + @Override + public SaBaseClientLoginUser createClientUserWithPhone(String phone) { + ClientUser clientUser = clientUserService.createUserWithPhone(phone); + return BeanUtil.copyProperties(clientUser, SaBaseClientLoginUser.class); + } + + @Override + public SaBaseLoginUser createUserWithEmail(String email) { + return null; + } + + @Override + public SaBaseClientLoginUser createClientUserWithEmail(String email) { + ClientUser clientUser = clientUserService.createUserWithEmail(email); + return BeanUtil.copyProperties(clientUser, SaBaseClientLoginUser.class); + } + + @Override + public void doRegister(String account, String password) { + clientUserService.doRegister(account, password); + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/result/ClientUserPicValidCodeResult.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/result/ClientUserPicValidCodeResult.java new file mode 100644 index 00000000..0bcacc7f --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/result/ClientUserPicValidCodeResult.java @@ -0,0 +1,36 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.result; + +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; + +/** + * 图片验证码结果 + * + * @author xuyuxiang + * @date 2022/7/8 9:28 + **/ +@Getter +@Setter +public class ClientUserPicValidCodeResult { + + /** 验证码图片,Base64 */ + @Schema(description = "验证码图片,Base64") + private String validCodeBase64; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + private String validCodeReqNo; +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserExtService.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserExtService.java new file mode 100644 index 00000000..044f5c34 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserExtService.java @@ -0,0 +1,41 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.client.modular.user.entity.ClientUserExt; + +/** + * C端用户扩展Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface ClientUserExtService extends IService { + + /** + * 更新用户最新修改密码时间 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void updatePasswordLastTime(String userId); + + /** + * 插入扩展信息 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void createExtInfo(String userId, String sourceFromType); +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserPasswordService.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserPasswordService.java new file mode 100644 index 00000000..82eb1ccb --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserPasswordService.java @@ -0,0 +1,43 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.client.modular.user.entity.ClientUserPassword; + +import java.util.List; + +/** + * C端用户密码Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface ClientUserPasswordService extends IService { + + /** + * 追加用户历史密码信息 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void insertUserPasswordHistory(String userId, String newPassword); + + /** + * 获取用户前N个历史密码 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + List getUserPasswordHistoryLimit(String userId, int limitValue); +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserService.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserService.java index 6bd00084..66022b3d 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserService.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/ClientUserService.java @@ -14,17 +14,16 @@ package vip.xiaonuo.client.modular.user.service; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.IService; +import org.springframework.web.multipart.MultipartFile; import vip.xiaonuo.client.modular.user.entity.ClientUser; -import vip.xiaonuo.client.modular.user.param.ClientUserAddParam; -import vip.xiaonuo.client.modular.user.param.ClientUserEditParam; -import vip.xiaonuo.client.modular.user.param.ClientUserIdParam; -import vip.xiaonuo.client.modular.user.param.ClientUserPageParam; +import vip.xiaonuo.client.modular.user.param.*; import vip.xiaonuo.client.modular.user.result.ClientLoginUser; +import vip.xiaonuo.client.modular.user.result.ClientUserPicValidCodeResult; import java.util.List; /** - * C端用户Service接口 + * 用户Service接口 * * @author xuyuxiang * @date 2022/4/21 18:35 @@ -64,7 +63,7 @@ public interface ClientUserService extends IService { ClientLoginUser getUserByEmail(String email); /** - * 获取C端用户分页 + * 获取用户分页 * * @author xuyuxiang * @date 2022/4/24 20:08 @@ -72,15 +71,15 @@ public interface ClientUserService extends IService { Page page(ClientUserPageParam clientUserPageParam); /** - * 添加C端用户 + * 添加用户 * * @author xuyuxiang * @date 2022/4/24 20:48 */ - void add(ClientUserAddParam clientUserAddParam); + void add(ClientUserAddParam clientUserAddParam, String sourceFromType); /** - * 编辑C端用户 + * 编辑用户 * * @author xuyuxiang * @date 2022/4/24 21:13 @@ -88,7 +87,7 @@ public interface ClientUserService extends IService { void edit(ClientUserEditParam clientUserEditParam); /** - * 删除C端用户 + * 删除用户 * * @author xuyuxiang * @date 2022/4/24 21:18 @@ -96,7 +95,7 @@ public interface ClientUserService extends IService { void delete(List clientUserIdParamList); /** - * 获取C端用户详情 + * 获取用户详情 * * @author xuyuxiang * @date 2022/4/24 21:18 @@ -104,7 +103,159 @@ public interface ClientUserService extends IService { ClientUser detail(ClientUserIdParam clientUserIdParam); /** - * 更新C端用户的登录时间和登录ip等信息 + * 获取用户详情 + * + * @author xuyuxiang + * @date 2022/4/24 21:18 + */ + ClientUser queryEntity(String id); + + /** + * 获取图片验证码 + * + * @author xuyuxiang + * @date 2021/12/28 14:46 + **/ + ClientUserPicValidCodeResult getPicCaptcha(); + + /** + * 找回密码获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String findPasswordGetPhoneValidCode(ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam); + + /** + * 找回密码获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String findPasswordGetEmailValidCode(ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam); + + /** + * 通过手机号找回用户密码 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + void findPasswordByPhone(ClientUserFindPwdByPhoneParam clientUserFindPwdByPhoneParam); + + /** + * 通过邮箱找回用户密码 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + void findPasswordByEmail(ClientUserFindPwdByEmailParam clientUserFindPwdByEmailParam); + + /** + * 修改密码获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updatePasswordGetPhoneValidCode(ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam); + + /** + * 修改密码获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updatePasswordGetEmailValidCode(ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam); + + /** + * 通过验证旧密码修改用户密码 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + void updatePasswordByOld(ClientUserUpdatePwdByOldParam clientUserUpdatePwdByOldParam); + + /** + * 通过验证手机号修改用户密码 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + void updatePasswordByPhone(ClientUserUpdatePwdByPhoneParam clientUserUpdatePwdByPhoneParam); + + /** + * 通过验证邮箱修改用户密码 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + void updatePasswordByEmail(ClientUserUpdatePwdByEmailParam clientUserUpdatePwdByEmailParam); + + /** + * 绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String bindPhoneGetPhoneValidCode(ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam); + + /** + * 修改绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updateBindPhoneGetPhoneValidCode(ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam); + + /** + * 绑定手机号 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void bindPhone(ClientUserBindPhoneParam clientUserBindPhoneParam); + + /** + * 绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String bindEmailGetEmailValidCode(ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam); + + /** + * 修改绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updateBindEmailGetEmailValidCode(ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam); + + /** + * 绑定邮箱 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void bindEmail(ClientUserBindEmailParam clientUserBindEmailParam); + + /** + * 修改用户头像返回base64 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + String updateAvatar(MultipartFile file); + + /** + * 修改用户签名图片返回base64 + * + * @author xuyuxiang yubaoshan + * @date 2022/4/22 15:53 + **/ + void updateSignature(ClientUserSignatureParam clientUserSignatureParam); + + /** + * 更新用户的登录时间和登录ip等信息 * * @author xuyuxiang * @date 2022/4/27 22:58 @@ -112,10 +263,82 @@ public interface ClientUserService extends IService { void updateUserLoginInfo(String userId, String device); /** - * 获取C端用户详情 + * 编辑个人信息 * * @author xuyuxiang - * @date 2022/4/24 21:18 + * @date 2022/4/24 20:47 */ - ClientUser queryEntity(String id); + void updateUserInfo(ClientUserUpdateInfoParam clientUserUpdateInfoParam); + + /** + * 根据id获取头像 + * + * @author xuyuxiang + * @date 2023/8/28 10:10 + **/ + String getAvatarById(ClientUserIdParam clientUserIdParam); + + /** + * 根据手机号创建用户 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + ClientUser createUserWithPhone(String phone); + + /** + * 根据邮箱创建用户 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + ClientUser createUserWithEmail(String email); + + /** + * 根据账号密码创建用户 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + ClientUser createUserWithAccount(String account, String password); + + /** + * 判断当前用户密码是否过期 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + Boolean isUserPasswordExpired(); + + /** + * 判断当前用户是否需要绑定手机号 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + Boolean isUserNeedBindPhone(); + + /** + * 判断当前用户是否需要绑定邮箱 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + Boolean isUserNeedBindEmail(); + + /** + * 通知用户密码即将到期 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void noticeUserPasswordAboutToExpired(); + + /** + * 执行注册 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void doRegister(String account, String password); } diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserExtServiceImpl.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserExtServiceImpl.java new file mode 100644 index 00000000..16745d30 --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserExtServiceImpl.java @@ -0,0 +1,57 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.service.impl; + +import cn.hutool.core.date.DateTime; +import cn.hutool.core.util.ObjectUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import vip.xiaonuo.client.modular.user.entity.ClientUserExt; +import vip.xiaonuo.client.modular.user.enums.ClientUserSourceFromTypeEnum; +import vip.xiaonuo.client.modular.user.mapper.ClientUserExtMapper; +import vip.xiaonuo.client.modular.user.service.ClientUserExtService; + +/** + * C端用户扩展Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class ClientUserExtServiceImpl extends ServiceImpl implements ClientUserExtService { + + @Override + public void updatePasswordLastTime(String userId) { + ClientUserExt clientUserExt = this.getOne(new LambdaQueryWrapper().eq(ClientUserExt::getUserId, userId)); + if(ObjectUtil.isEmpty(clientUserExt)){ + clientUserExt = new ClientUserExt(); + clientUserExt.setUserId(userId); + clientUserExt.setSourceFromType(ClientUserSourceFromTypeEnum.SYSTEM_ADD.getValue()); + clientUserExt.setPasswordUpdateTime(DateTime.now()); + this.save(clientUserExt); + } else { + clientUserExt.setPasswordUpdateTime(DateTime.now()); + this.updateById(clientUserExt); + } + } + + @Override + public void createExtInfo(String userId, String sourceFromType) { + ClientUserExt clientUserExt = new ClientUserExt(); + clientUserExt.setUserId(userId); + clientUserExt.setSourceFromType(sourceFromType); + clientUserExt.setPasswordUpdateTime(DateTime.now()); + this.save(clientUserExt); + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserPasswordServiceImpl.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserPasswordServiceImpl.java new file mode 100644 index 00000000..c45377ee --- /dev/null +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserPasswordServiceImpl.java @@ -0,0 +1,50 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.client.modular.user.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import vip.xiaonuo.client.modular.user.entity.ClientUserPassword; +import vip.xiaonuo.client.modular.user.mapper.ClientUserPasswordMapper; +import vip.xiaonuo.client.modular.user.service.ClientUserPasswordService; +import vip.xiaonuo.common.util.CommonCryptogramUtil; + +import java.util.List; +import java.util.stream.Collectors; + +/** + * C端用户密码Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class ClientUserPasswordServiceImpl extends ServiceImpl implements ClientUserPasswordService { + + @Override + public void insertUserPasswordHistory(String userId, String newPassword) { + ClientUserPassword clientUserPassword = new ClientUserPassword(); + clientUserPassword.setUserId(userId); + clientUserPassword.setPassword(CommonCryptogramUtil.doHashValue(newPassword)); + this.save(clientUserPassword); + } + + @Override + public List getUserPasswordHistoryLimit(String userId, int limitValue) { + return this.page(new Page<>(1, limitValue), new LambdaQueryWrapper() + .eq(ClientUserPassword::getUserId, userId).orderByDesc(ClientUserPassword::getCreateTime)) + .getRecords().stream().map(ClientUserPassword::getPassword).collect(Collectors.toList()); + } +} diff --git a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserServiceImpl.java b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserServiceImpl.java index a93ab1f1..72c44791 100644 --- a/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserServiceImpl.java +++ b/snowy-plugin/snowy-plugin-client/src/main/java/vip/xiaonuo/client/modular/user/service/impl/ClientUserServiceImpl.java @@ -12,37 +12,60 @@ */ package vip.xiaonuo.client.modular.user.service.impl; +import cn.hutool.captcha.CaptchaUtil; +import cn.hutool.captcha.CircleCaptcha; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.collection.CollStreamUtil; +import cn.hutool.core.convert.Convert; import cn.hutool.core.date.DateTime; +import cn.hutool.core.img.ImgUtil; +import cn.hutool.core.io.FileUtil; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.PhoneUtil; +import cn.hutool.core.util.RandomUtil; import cn.hutool.core.util.StrUtil; +import cn.hutool.json.JSONObject; +import cn.hutool.json.JSONUtil; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper; +import com.baomidou.mybatisplus.core.toolkit.IdWorker; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.fhs.trans.service.impl.TransService; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; -import vip.xiaonuo.client.modular.relation.service.ClientRelationService; +import org.springframework.web.multipart.MultipartFile; +import vip.xiaonuo.auth.core.util.StpClientUtil; +import vip.xiaonuo.auth.core.util.StpLoginUserUtil; +import vip.xiaonuo.client.core.util.ClientEmailFormatUtl; +import vip.xiaonuo.client.core.util.ClientPasswordUtl; import vip.xiaonuo.client.modular.user.entity.ClientUser; +import vip.xiaonuo.client.modular.user.entity.ClientUserExt; +import vip.xiaonuo.client.modular.user.enums.ClientUpdatePasswordValidTypeEnum; +import vip.xiaonuo.client.modular.user.enums.ClientUserSourceFromTypeEnum; import vip.xiaonuo.client.modular.user.enums.ClientUserStatusEnum; import vip.xiaonuo.client.modular.user.mapper.ClientUserMapper; -import vip.xiaonuo.client.modular.user.param.ClientUserAddParam; -import vip.xiaonuo.client.modular.user.param.ClientUserEditParam; -import vip.xiaonuo.client.modular.user.param.ClientUserIdParam; -import vip.xiaonuo.client.modular.user.param.ClientUserPageParam; +import vip.xiaonuo.client.modular.user.param.*; import vip.xiaonuo.client.modular.user.result.ClientLoginUser; +import vip.xiaonuo.client.modular.user.result.ClientUserPicValidCodeResult; +import vip.xiaonuo.client.modular.user.service.ClientUserExtService; +import vip.xiaonuo.client.modular.user.service.ClientUserPasswordService; import vip.xiaonuo.client.modular.user.service.ClientUserService; +import vip.xiaonuo.common.cache.CommonCacheOperator; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.page.CommonPageRequest; import vip.xiaonuo.common.util.*; import vip.xiaonuo.dev.api.DevConfigApi; +import vip.xiaonuo.dev.api.DevEmailApi; +import vip.xiaonuo.dev.api.DevSmsApi; +import java.awt.image.BufferedImage; +import java.io.IOException; import java.util.List; +import java.util.Objects; /** * C端用户Service接口实现类 @@ -53,16 +76,80 @@ import java.util.List; @Service public class ClientUserServiceImpl extends ServiceImpl implements ClientUserService { - private static final String SNOWY_SYS_DEFAULT_PASSWORD_KEY = "SNOWY_SYS_DEFAULT_PASSWORD"; + /** C端验证码失效时间(适用图片验证码和短信验证码,单位:分钟,默认5分钟有效) */ + private static final String SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_C_KEY = "SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_C"; + + /** C端重置密码验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_C"; + + /** C端重置密码验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_C"; + + /** C端修改密码验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C"; + + /** C端修改密码验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C"; + + /** C端重置密码成功短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C"; + + /** C端重置密码成功邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C"; + + /** C端密码即将到期短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C"; + + /** C端密码即将到期邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C"; + + /** C端绑定手机验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_C"; + + /** C端绑定邮箱验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_C"; + + /** C端修改绑定手机验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_C"; + + /** C端修改绑定邮箱验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_C"; + + /** C端注册账号成功短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C_KEY = "SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C"; + + /** C端注册账号成功邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C_KEY = "SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C"; + + /** C端注册后是否需要绑定手机号 */ + private static final String SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_C_KEY = "SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_C"; + + /** C端注册后是否需要绑定邮箱 */ + private static final String SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_C_KEY = "SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_C"; + + /** 验证码缓存前缀 */ + private static final String USER_VALID_CODE_CACHE_KEY = "user-validCode:"; @Resource - private DevConfigApi devConfigApi; + private CommonCacheOperator commonCacheOperator; + + @Resource + private DevSmsApi devSmsApi; @Resource private TransService transService; @Resource - private ClientRelationService clientRelationService; + private DevEmailApi devEmailApi; + + @Resource + private DevConfigApi devConfigApi; + + @Resource + private ClientUserExtService clientUserExtService; + + @Resource + private ClientUserPasswordService clientUserPasswordService; @Override public ClientLoginUser getUserById(String id) { @@ -123,18 +210,26 @@ public class ClientUserServiceImpl extends ServiceImpl() @@ -187,7 +282,7 @@ public class ClientUserServiceImpl extends ServiceImpl() @@ -224,6 +319,35 @@ public class ClientUserServiceImpl extends ServiceImpl lambdaUpdateWrapper = new LambdaUpdateWrapper().eq(ClientUser::getId, clientUser.getId()); + if(ObjectUtil.isNotEmpty(clientUserUpdateInfoParam.getName())) { + lambdaUpdateWrapper.set(ClientUser::getName, clientUserUpdateInfoParam.getName()); + } + if(ObjectUtil.isNotEmpty(clientUserUpdateInfoParam.getNickname())) { + lambdaUpdateWrapper.set(ClientUser::getNickname, clientUserUpdateInfoParam.getNickname()); + } + if(ObjectUtil.isNotEmpty(clientUserUpdateInfoParam.getGender())) { + lambdaUpdateWrapper.set(ClientUser::getGender, clientUserUpdateInfoParam.getGender()); + } + if(ObjectUtil.isNotEmpty(clientUserUpdateInfoParam.getBirthday())) { + lambdaUpdateWrapper.set(ClientUser::getBirthday, clientUserUpdateInfoParam.getBirthday()); + } + // 更新指定字段 + this.update(lambdaUpdateWrapper); + } + + @Override + public String getAvatarById(ClientUserIdParam clientUserIdParam) { + return this.detail(clientUserIdParam).getAvatar(); + } + @Override public ClientUser queryEntity(String id) { ClientUser clientUser = this.getById(id); @@ -232,4 +356,792 @@ public class ClientUserServiceImpl extends ServiceImpl().eq(ClientUser::getPhone, + CommonCryptogramUtil.doSm4CbcEncrypt(phone)).set(ClientUser::getPassword, + CommonCryptogramUtil.doHashValue(newPassword))); + // 更新用户最新修改密码时间 + clientUserExtService.updatePasswordLastTime(clientLoginUser.getId()); + // 追加用户历史密码信息 + clientUserPasswordService.insertUserPasswordHistory(clientLoginUser.getId(), newPassword); + // 重置密码成功短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("userNewPassword", newPassword); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } + } + + @Override + public void findPasswordByEmail(ClientUserFindPwdByEmailParam clientUserFindPwdByEmailParam) { + // 再次校验邮箱是否合法 + String email = clientUserFindPwdByEmailParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 根据邮箱获取用户信息,判断用户是否存在 + ClientLoginUser clientLoginUser = this.getUserByEmail(email); + if (ObjectUtil.isEmpty(clientLoginUser)) { + throw new CommonException("邮箱:{}不存在对应用户", email); + } + // 执行校验验证码 + validValidCode(email, clientUserFindPwdByEmailParam.getValidCode(), clientUserFindPwdByEmailParam.getValidCodeReqNo()); + // 获取新密码 + String newPassword = CommonCryptogramUtil.doSm2Decrypt(clientUserFindPwdByEmailParam.getNewPassword()).trim(); + // 校验新密码 + ClientPasswordUtl.validNewPassword(clientLoginUser, newPassword); + // 修改密码 + this.update(new LambdaUpdateWrapper().eq(ClientUser::getEmail, email).set(ClientUser::getPassword, + CommonCryptogramUtil.doHashValue(newPassword))); + // 更新用户最新修改密码时间 + clientUserExtService.updatePasswordLastTime(clientLoginUser.getId()); + // 追加用户历史密码信息 + clientUserPasswordService.insertUserPasswordHistory(clientLoginUser.getId(), newPassword); + // 重置密码成功邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", clientUserFindPwdByEmailParam.getEmail()).set("userNewPassword", newPassword); + // 获取格式化后的主题 + String subject = ClientEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = ClientEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } + } + + @Override + public String updatePasswordGetPhoneValidCode(ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam) { + // 判断密码验证方式 + ClientPasswordUtl.validUpdatePasswordValidType(ClientUpdatePasswordValidTypeEnum.PHONE.getValue()); + return this.getPhoneValidCode(clientUserGetPhoneValidCodeParam, "修改密码验证码短信消息模板编码", SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C_KEY); + } + + @Override + public String updatePasswordGetEmailValidCode(ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam) { + // 判断密码验证方式 + ClientPasswordUtl.validUpdatePasswordValidType(ClientUpdatePasswordValidTypeEnum.EMAIL.getValue()); + return this.getEmailValidCode(clientUserGetEmailValidCodeParam, "修改密码验证码邮件消息模板内容", SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C_KEY); + } + + @Override + public void updatePasswordByOld(ClientUserUpdatePwdByOldParam clientUserUpdatePwdByOldParam) { + // 判断密码验证方式 + ClientPasswordUtl.validUpdatePasswordValidType(ClientUpdatePasswordValidTypeEnum.OLD.getValue()); + ClientUser clientUser = this.queryEntity(StpClientUtil.getLoginIdAsString()); + String password = CommonCryptogramUtil.doSm2Decrypt(clientUserUpdatePwdByOldParam.getPassword()).trim(); + String newPassword = CommonCryptogramUtil.doSm2Decrypt(clientUserUpdatePwdByOldParam.getNewPassword()).trim(); + if (!CommonCryptogramUtil.doHashValue(password).equals(clientUser.getPassword())) { + throw new CommonException("原密码错误"); + } + // 校验新密码 + ClientPasswordUtl.validNewPassword(clientUser, newPassword); + // 修改密码 + this.update(new LambdaUpdateWrapper().eq(ClientUser::getId, + clientUser.getId()).set(ClientUser::getPassword, + CommonCryptogramUtil.doHashValue(newPassword))); + // 更新用户最新修改密码时间 + clientUserExtService.updatePasswordLastTime(clientUser.getId()); + // 追加用户历史密码信息 + clientUserPasswordService.insertUserPasswordHistory(clientUser.getId(), newPassword); + // 获取手机号 + String phone = clientUser.getPhone(); + // 手机号不为空则发送密码重置成功短信 + if(ObjectUtil.isNotEmpty(phone)){ + // 重置密码成功短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("userNewPassword", newPassword); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 编码不为空 + if(ObjectUtil.isNotEmpty(codeValue)){ + try { + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } catch (Exception e) { + log.error(">>> 短信发送失败", e); + } + } + } + } + // 获取手机号 + String email = clientUser.getEmail(); + // 密码不为空则发送密码重置成功邮件 + if(ObjectUtil.isNotEmpty(email)){ + // 重置密码成功邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("userNewPassword", newPassword); + // 获取格式化后的主题 + String subject = ClientEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = ClientEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + try { + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } catch (Exception e) { + log.error(">>> 邮件发送失败", e); + } + } + } + } + + @Override + public void updatePasswordByPhone(ClientUserUpdatePwdByPhoneParam clientUserUpdatePwdByPhoneParam) { + // 判断密码验证方式 + ClientPasswordUtl.validUpdatePasswordValidType(ClientUpdatePasswordValidTypeEnum.PHONE.getValue()); + ClientUserFindPwdByPhoneParam clientUserFindPwdByPhoneParam = new ClientUserFindPwdByPhoneParam(); + BeanUtil.copyProperties(clientUserUpdatePwdByPhoneParam, clientUserFindPwdByPhoneParam); + this.findPasswordByPhone(clientUserFindPwdByPhoneParam); + } + + @Override + public void updatePasswordByEmail(ClientUserUpdatePwdByEmailParam clientUserUpdatePwdByEmailParam) { + // 判断密码验证方式 + ClientPasswordUtl.validUpdatePasswordValidType(ClientUpdatePasswordValidTypeEnum.EMAIL.getValue()); + ClientUserFindPwdByEmailParam clientUserFindPwdByEmailParam = new ClientUserFindPwdByEmailParam(); + BeanUtil.copyProperties(clientUserUpdatePwdByEmailParam, clientUserFindPwdByEmailParam); + this.findPasswordByEmail(clientUserFindPwdByEmailParam); + } + + @Override + public String bindPhoneGetPhoneValidCode(ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam) { + // 手机号 + String phone = clientUserGetPhoneValidCodeParam.getPhone(); + // 校验手机号格式 + if (!PhoneUtil.isMobile(phone)) { + throw new CommonException("手机号码:{}格式错误", phone); + } + // 执行校验验证码 + validValidCode(null, clientUserGetPhoneValidCodeParam.getValidCode(), clientUserGetPhoneValidCodeParam.getValidCodeReqNo()); + // 根据手机号获取用户信息,判断用户是否存在,如果存在则不能绑定该手机号 + if (ObjectUtil.isNotEmpty(this.getUserByPhone(phone))) { + throw new CommonException("手机号码:{}已存在对应用户", phone); + } + // 生成手机验证码的值,随机6为数字 + String phoneValidCode = RandomUtil.randomNumbers(6); + // 生成手机验证码的请求号 + String phoneValidCodeReqNo = IdWorker.getIdStr(); + // 绑定手机验证码短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_C_KEY); + if(ObjectUtil.isEmpty(smsTemplateCode)){ + throw new CommonException("请联系管理员配置绑定手机验证码短信消息模板编码"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("validCode", phoneValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, validCodeExpiredDuration); + // 返回请求号 + return phoneValidCodeReqNo; + } + + @Override + public String updateBindPhoneGetPhoneValidCode(ClientUserGetPhoneValidCodeParam clientUserGetPhoneValidCodeParam) { + // 手机号 + String phone = clientUserGetPhoneValidCodeParam.getPhone(); + // 校验手机号格式 + if (!PhoneUtil.isMobile(phone)) { + throw new CommonException("手机号码:{}格式错误", phone); + } + // 执行校验验证码 + validValidCode(null, clientUserGetPhoneValidCodeParam.getValidCode(), clientUserGetPhoneValidCodeParam.getValidCodeReqNo()); + // 根据手机号获取用户信息,判断用户是否存在,如果存在则不能绑定该手机号 + if (ObjectUtil.isNotEmpty(this.getUserByPhone(phone))) { + throw new CommonException("手机号码:{}已存在对应用户", phone); + } + // 生成手机验证码的值,随机6为数字 + String phoneValidCode = RandomUtil.randomNumbers(6); + // 生成手机验证码的请求号 + String phoneValidCodeReqNo = IdWorker.getIdStr(); + // 修改绑定手机验证码短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_C_KEY); + if(ObjectUtil.isEmpty(smsTemplateCode)){ + throw new CommonException("请联系管理员配置修改绑定手机验证码短信消息模板编码"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("validCode", phoneValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, validCodeExpiredDuration); + // 返回请求号 + return phoneValidCodeReqNo; + } + + @Override + public void bindPhone(ClientUserBindPhoneParam clientUserBindPhoneParam) { + // 再次校验手机号是否合法 + String phone = clientUserBindPhoneParam.getPhone(); + // 校验手机号格式 + if (!PhoneUtil.isMobile(phone)) { + throw new CommonException("手机号码:{}格式错误", phone); + } + // 根据手机号获取用户信息,判断用户是否存在,如果存在则不能绑定该手机号 + if (ObjectUtil.isNotEmpty(this.getUserByPhone(phone))) { + throw new CommonException("手机号码:{}已存在对应用户", phone); + } + // 执行校验验证码 + validValidCode(phone, clientUserBindPhoneParam.getValidCode(), clientUserBindPhoneParam.getValidCodeReqNo()); + // 修改手机号 + this.update(new LambdaUpdateWrapper().eq(ClientUser::getId, StpClientUtil.getLoginIdAsString()) + .set(ClientUser::getPhone, CommonCryptogramUtil.doSm4CbcEncrypt(phone))); + } + + @Override + public String bindEmailGetEmailValidCode(ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam) { + // 邮箱 + String email = clientUserGetEmailValidCodeParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 执行校验验证码 + validValidCode(null, clientUserGetEmailValidCodeParam.getValidCode(), clientUserGetEmailValidCodeParam.getValidCodeReqNo()); + // 根据邮箱获取用户信息,判断用户是否存在,如果存在则不能绑定该邮箱 + if (ObjectUtil.isEmpty(this.getUserByEmail(email))) { + throw new CommonException("邮箱:{}已存在对应用户", email); + } + // 生成邮箱验证码的值,随机6为数字 + String emailValidCode = RandomUtil.randomNumbers(6); + // 生成邮箱验证码的请求号 + String emailValidCodeReqNo = IdWorker.getIdStr(); + // 绑定邮箱验证码邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_C_KEY); + if(ObjectUtil.isEmpty(emailTemplateContent)){ + throw new CommonException("请联系管理员配置绑定邮箱验证码邮件消息模板内容"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("validCode", emailValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取格式化后的主题 + String subject = ClientEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = ClientEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + email + StrUtil.UNDERLINE + emailValidCodeReqNo, emailValidCode, validCodeExpiredDuration); + // 返回请求号 + return emailValidCodeReqNo; + } + + @Override + public String updateBindEmailGetEmailValidCode(ClientUserGetEmailValidCodeParam clientUserGetEmailValidCodeParam) { + // 邮箱 + String email = clientUserGetEmailValidCodeParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 执行校验验证码 + validValidCode(null, clientUserGetEmailValidCodeParam.getValidCode(), clientUserGetEmailValidCodeParam.getValidCodeReqNo()); + // 根据邮箱获取用户信息,判断用户是否存在,如果存在则不能绑定该邮箱 + if (ObjectUtil.isEmpty(this.getUserByEmail(email))) { + throw new CommonException("邮箱:{}已存在对应用户", email); + } + // 生成邮箱验证码的值,随机6为数字 + String emailValidCode = RandomUtil.randomNumbers(6); + // 生成邮箱验证码的请求号 + String emailValidCodeReqNo = IdWorker.getIdStr(); + // 修改绑定邮箱验证码邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_C_KEY); + if(ObjectUtil.isEmpty(emailTemplateContent)){ + throw new CommonException("请联系管理员配置修改绑定邮箱验证码邮件消息模板内容"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("validCode", emailValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取格式化后的主题 + String subject = ClientEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = ClientEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + email + StrUtil.UNDERLINE + emailValidCodeReqNo, emailValidCode, validCodeExpiredDuration); + // 返回请求号 + return emailValidCodeReqNo; + } + + @Override + public void bindEmail(ClientUserBindEmailParam clientUserBindEmailParam) { + // 再次校验邮箱是否合法 + String email = clientUserBindEmailParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 根据邮箱获取用户信息,判断用户是否存在,如果存在则不能绑定该邮箱 + if (ObjectUtil.isEmpty(this.getUserByEmail(email))) { + throw new CommonException("邮箱:{}已存在对应用户", email); + } + // 执行校验验证码 + validValidCode(email, clientUserBindEmailParam.getValidCode(), clientUserBindEmailParam.getValidCodeReqNo()); + // 修改邮箱 + this.update(new LambdaUpdateWrapper().eq(ClientUser::getId, StpClientUtil.getLoginIdAsString()) + .set(ClientUser::getEmail, email)); + } + + @Override + public String updateAvatar(MultipartFile file) { + ClientUser clientUser = this.queryEntity(StpClientUtil.getLoginIdAsString()); + try { + String suffix = Objects.requireNonNull(FileUtil.getSuffix(file.getOriginalFilename())).toLowerCase(); + BufferedImage image = ImgUtil.toImage(file.getBytes()); + String base64; + if(image.getWidth() <= 200 && image.getHeight() <= 200) { + base64 = ImgUtil.toBase64DataUri(image, suffix); + } else { + base64 = ImgUtil.toBase64DataUri(ImgUtil.scale(image, 200, 200, null), suffix); + } + this.update(new LambdaUpdateWrapper().eq(ClientUser::getId, + clientUser.getId()).set(ClientUser::getAvatar, base64)); + return base64; + } catch (IOException e) { + log.error(">>> 头像修改失败:", e); + throw new CommonException("头像修改失败,用户id值为:{}", clientUser.getId()); + } + } + + @Override + public void updateSignature(ClientUserSignatureParam clientUserSignatureParam) { + ClientUser clientUser = this.queryEntity(StpClientUtil.getLoginIdAsString()); + String clientUserSignatureStr = clientUserSignatureParam.getSignature(); + if(clientUserSignatureParam.getSignature().contains(StrUtil.COMMA)) { + clientUserSignatureStr = StrUtil.split(clientUserSignatureStr, StrUtil.COMMA).get(1); + } + String base64 = ImgUtil.toBase64DataUri(ImgUtil.scale(ImgUtil.toImage(clientUserSignatureStr), + 100, 50, null), ImgUtil.IMAGE_TYPE_PNG); + // 更新指定字段 + this.update(new LambdaUpdateWrapper().eq(ClientUser::getId, clientUser.getId()) + .set(ClientUser::getSignature, base64)); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public ClientUser createUserWithPhone(String phone) { + ClientUserAddParam clientUserAddParam = new ClientUserAddParam(); + clientUserAddParam.setAccount(phone); + clientUserAddParam.setName(phone); + clientUserAddParam.setPhone(phone); + // 保存用户 + this.add(clientUserAddParam, ClientUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue()); + // 获取用户信息 + ClientUser clientUser = this.getOne(new LambdaQueryWrapper().eq(ClientUser::getPhone, CommonCryptogramUtil.doSm4CbcEncrypt(phone))); + // 发送注册成功短信 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } + // 返回用户 + return clientUser; + } + + @Transactional(rollbackFor = Exception.class) + @Override + public ClientUser createUserWithEmail(String email) { + ClientUserAddParam clientUserAddParam = new ClientUserAddParam(); + clientUserAddParam.setAccount(email); + clientUserAddParam.setName(email); + clientUserAddParam.setEmail(email); + // 保存用户 + this.add(clientUserAddParam, ClientUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue()); + // 获取用户信息 + ClientUser clientUser = this.getOne(new LambdaQueryWrapper().eq(ClientUser::getEmail, email)); + // 发送注册成功邮件 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(emailTemplateContent)) { + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email); + // 获取格式化后的主题 + String subject = ClientEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap); + // 获取格式化后的内容 + String content = ClientEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap); + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } + // 返回用户 + return clientUser; + } + + @Transactional(rollbackFor = Exception.class) + @Override + public ClientUser createUserWithAccount(String account, String password) { + ClientUserAddParam clientUserAddParam = new ClientUserAddParam(); + clientUserAddParam.setAccount(account); + clientUserAddParam.setName(account); + clientUserAddParam.setPassword(password); + // 保存用户 + this.add(clientUserAddParam, ClientUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue()); + // 返回用户 + return this.getOne(new LambdaQueryWrapper().eq(ClientUser::getAccount, account)); + } + + @Override + public Boolean isUserPasswordExpired() { + return ClientPasswordUtl.isUserPasswordExpired(StpClientUtil.getLoginIdAsString()); + } + + @Override + public Boolean isUserNeedBindPhone() { + // 获取当前用户 + ClientUser clientUser = this.queryEntity(StpClientUtil.getLoginIdAsString()); + // 查询当前用户是否注册的 + ClientUserExt clientUserExt = clientUserExtService.getOne(new LambdaQueryWrapper().eq(ClientUserExt::getUserId, StpClientUtil.getLoginIdAsString()) + .eq(ClientUserExt::getSourceFromType, ClientUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue())); + // 不为空,则判断手机号是否为空 + if(ObjectUtil.isNotEmpty(clientUserExt)){ + // 手机号为空,判断系统注册后是否需要绑定手机号 + if(ObjectUtil.isEmpty(clientUser.getPhone())) { + String registerNeedBindPhone = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(registerNeedBindPhone)){ + return Convert.toBool(registerNeedBindPhone); + } else { + return false; + } + } else { + return false; + } + } else { + return false; + } + } + + @Override + public Boolean isUserNeedBindEmail() { + // 获取当前用户 + ClientUser clientUser = this.queryEntity(StpClientUtil.getLoginIdAsString()); + // 查询当前用户是否注册的 + ClientUserExt clientUserExt = clientUserExtService.getOne(new LambdaQueryWrapper().eq(ClientUserExt::getUserId, StpClientUtil.getLoginIdAsString()) + .eq(ClientUserExt::getSourceFromType, ClientUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue())); + // 不为空,则判断邮箱是否为空 + if(ObjectUtil.isNotEmpty(clientUserExt)){ + // 邮箱为空,判断系统注册后是否需要绑定邮箱 + if(ObjectUtil.isEmpty(clientUser.getEmail())) { + String registerNeedBindEmail = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_C_KEY); + if(ObjectUtil.isNotEmpty(registerNeedBindEmail)){ + return Convert.toBool(registerNeedBindEmail); + } else { + return false; + } + } else { + return false; + } + } else { + return false; + } + } + + @Override + public void noticeUserPasswordAboutToExpired() { + // 密码即将到期短信消息模板 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C_KEY); + // 密码即将到期邮件消息模板 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C_KEY); + // 获取今日需要提醒密码到期的用户集合 + ClientPasswordUtl.thisDayPasswordExpiredNeedNoticeUserIdList().forEach(clientUser -> { + // 获取手机号 + String phone = clientUser.getPhone(); + // 不为空才发送 + if(ObjectUtil.isAllNotEmpty(phone, smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 编码不为空 + if(ObjectUtil.isNotEmpty(codeValue)){ + try { + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } catch (Exception e) { + log.error(">>> 短信发送失败", e); + } + } + } + // 获取邮箱 + String email = clientUser.getEmail(); + // 不为空才发送 + if(ObjectUtil.isAllNotEmpty(email, emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email); + // 获取格式化后的主题 + String subject = ClientEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = ClientEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + try { + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } catch (Exception e) { + log.error(">>> 邮件发送失败", e); + } + } + }); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void doRegister(String account, String password) { + // 校验账号 + ClientLoginUser clientLoginUser = this.getUserByAccount(account); + if(ObjectUtil.isNotEmpty(clientLoginUser)) { + throw new CommonException("账号已存在"); + } + // 校验密码 + ClientPasswordUtl.validNewPassword(password); + // 根据账号密码创建用户 + this.createUserWithAccount(account, password); + } + + /** + * 获取验证码失效时间(单位:秒) + * + * @author xuyuxiang + * @date 2025/3/21 20:25 + **/ + private long getValidCodeExpiredDuration() { + // 默认5分钟 + int defaultExpiredTime = 5; + // 获取配置验证码失效时间 + String configCaptchaExpiredDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_C_KEY); + // 判断是否为空 + if(ObjectUtil.isNotEmpty(configCaptchaExpiredDuration)){ + // 配置了则使用配置的失效时间 + defaultExpiredTime = Convert.toInt(configCaptchaExpiredDuration); + } + // 转为秒 + return defaultExpiredTime * 60L; + } } diff --git a/snowy-plugin/snowy-plugin-dev/pom.xml b/snowy-plugin/snowy-plugin-dev/pom.xml index 68f51eba..fc35ce48 100644 --- a/snowy-plugin/snowy-plugin-dev/pom.xml +++ b/snowy-plugin/snowy-plugin-dev/pom.xml @@ -15,6 +15,7 @@ 开发工具插件 + vip.xiaonuo @@ -33,4 +34,4 @@ snowy-plugin-sys-api - \ No newline at end of file + diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/core/listener/DevJobListener.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/core/listener/DevJobListener.java index dbf0ddac..36dbd0d8 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/core/listener/DevJobListener.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/core/listener/DevJobListener.java @@ -15,6 +15,7 @@ package vip.xiaonuo.dev.core.listener; import cn.hutool.cron.CronUtil; import cn.hutool.extra.spring.SpringUtil; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import lombok.NonNull; import lombok.extern.slf4j.Slf4j; import org.springframework.boot.context.event.ApplicationStartedEvent; import org.springframework.context.ApplicationListener; @@ -36,9 +37,8 @@ import vip.xiaonuo.dev.modular.job.service.DevJobService; @Configuration public class DevJobListener implements ApplicationListener, Ordered { - @SuppressWarnings("ALL") @Override - public void onApplicationEvent( ApplicationStartedEvent applicationStartedEvent) { + public void onApplicationEvent(@NonNull ApplicationStartedEvent applicationStartedEvent) { SpringUtil.getBean(DevJobService.class).list(new LambdaQueryWrapper() .eq(DevJob::getJobStatus, DevJobStatusEnum.RUNNING.getValue()).orderByAsc(DevJob::getSortCode)) .forEach(devJob -> CronUtil.schedule(devJob.getId(), devJob.getCronExpression(), () -> { @@ -52,7 +52,7 @@ public class DevJobListener implements ApplicationListener> page(DevConfigPageParam devConfigPageParam) { @@ -63,6 +81,7 @@ public class DevConfigController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取系统基础配置") @GetMapping("/dev/config/sysBaseList") public CommonResult> sysBaseList() { @@ -75,6 +94,7 @@ public class DevConfigController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取配置列表") @GetMapping("/dev/config/list") public CommonResult> list(DevConfigListParam devConfigListParam) { @@ -87,6 +107,7 @@ public class DevConfigController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "添加配置") @CommonLog("添加配置") @PostMapping("/dev/config/add") @@ -101,6 +122,7 @@ public class DevConfigController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "编辑配置") @CommonLog("编辑配置") @PostMapping("/dev/config/edit") @@ -115,6 +137,7 @@ public class DevConfigController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "删除配置") @CommonLog("删除配置") @PostMapping("/dev/config/delete") @@ -130,6 +153,7 @@ public class DevConfigController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取配置详情") @GetMapping("/dev/config/detail") public CommonResult detail(@Valid DevConfigIdParam devConfigIdParam) { @@ -142,6 +166,7 @@ public class DevConfigController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "配置批量更新") @CommonLog("配置批量更新") @PostMapping("/dev/config/editBatch") @@ -150,4 +175,55 @@ public class DevConfigController { devConfigService.editBatch(devConfigBatchParamList); return CommonResult.ok(); } + + /** + * 获取机构选树 + * + * @author yubaoshan + * @date 2025/4/23 20:00 + */ + @Operation(summary = "获取机构选树") + @GetMapping("/dev/config/orgTree") + public CommonResult>> orgTree() { + return CommonResult.data(sysOrgApi.orgTreeSelector()); + } + + /** + * 获取角色选择器 + * + * @author yubaoshan + * @date 2025/4/23 20:00 + */ + @Operation(summary = "获取角色选择器") + @GetMapping("/dev/config/roleSelector") + public CommonResult> roleSelector(DevConfigSelectorRoleParam devConfigSelectorRoleParam) { + return CommonResult.data(sysRoleApi.roleSelector(devConfigSelectorRoleParam.getOrgId(), devConfigSelectorRoleParam.getCategory(), + devConfigSelectorRoleParam.getSearchKey(), null , false)); + } + + /** + * 获取机构选择器 + * + * @author yubaoshan + * @date 2025/4/23 20:00 + */ + @Operation(summary = "获取机构选择器") + @GetMapping("/dev/config/orgSelector") + public CommonResult> orgSelector(DevConfigSelectorOrgListParam devConfigSelectorOrgListParam) { + return CommonResult.data(sysOrgApi.orgListSelector(devConfigSelectorOrgListParam.getParentId())); + } + + /** + * 获取职位选择器 + * + * @author yubaoshan + * @date 2025/4/23 20:00 + */ + @Operation(summary = "获取职位选择器") + @GetMapping("/dev/config/positionSelector") + public CommonResult> positionSelector(@Valid DevConfigSelectorPositionParam devConfigSelectorPositionParam) { + return CommonResult.data(sysPositionApi.positionSelector(devConfigSelectorPositionParam.getOrgId(), devConfigSelectorPositionParam.getSearchKey(), + devConfigSelectorPositionParam.getCurrent(), devConfigSelectorPositionParam.getSize())); + } + } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/entity/DevConfig.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/entity/DevConfig.java index 10d04717..fd907409 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/entity/DevConfig.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/entity/DevConfig.java @@ -57,6 +57,6 @@ public class DevConfig extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/enums/DevConfigCategoryEnum.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/enums/DevConfigCategoryEnum.java index 712d4336..05849d08 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/enums/DevConfigCategoryEnum.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/enums/DevConfigCategoryEnum.java @@ -90,9 +90,14 @@ public enum DevConfigCategoryEnum { SMS_ALIYUN("SMS_ALIYUN"), /** - * 短信-小诺短信 + * 支付-支付宝 */ - SMS_XIAONUO("SMS_XIAONUO"); + PAY_ALI("PAY_ALI"), + + /** + * 支付-微信 + */ + PAY_WX("PAY_WX"); private final String value; @@ -107,7 +112,7 @@ public enum DevConfigCategoryEnum { FILE_ALIYUN.getValue().equals(value) || FILE_MINIO.getValue().equals(value) || EMAIL_TENCENT.getValue().equals(value) || EMAIL_ALIYUN.getValue().equals(value) || SMS_TENCENT.getValue().equals(value) || SMS_ALIYUN.getValue().equals(value) || - SMS_XIAONUO.getValue().equals(value); + PAY_ALI.getValue().equals(value) || PAY_WX.getValue().equals(value); if(!flag) { throw new CommonException("不支持的配置分类:{}", value); } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/mapper/mapping/DevConfigMapper.xml b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/mapper/mapping/DevConfigMapper.xml index ba9a3918..7abd6474 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/mapper/mapping/DevConfigMapper.xml +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/mapper/mapping/DevConfigMapper.xml @@ -2,5 +2,4 @@ - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigAddParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigAddParam.java index 7f32f8a1..544c93c5 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigAddParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigAddParam.java @@ -29,12 +29,12 @@ import lombok.Setter; public class DevConfigAddParam { /** 配置键 */ - @Schema(description = "配置键", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "配置键") @NotBlank(message = "configKey不能为空") private String configKey; /** 配置值 */ - @Schema(description = "配置值", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "配置值") @NotBlank(message = "configValue不能为空") private String configValue; @@ -43,7 +43,7 @@ public class DevConfigAddParam { private String remark; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigBatchParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigBatchParam.java index 4c5c2252..2286d307 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigBatchParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigBatchParam.java @@ -28,12 +28,12 @@ import lombok.Setter; public class DevConfigBatchParam { /** 配置键 */ - @Schema(description = "配置键", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "配置键") @NotBlank(message = "configKey不能为空") private String configKey; /** 配置值 */ - @Schema(description = "配置值", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "配置值") @NotBlank(message = "configValue不能为空") private String configValue; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigEditParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigEditParam.java index f1009608..c66c8be5 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigEditParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigEditParam.java @@ -29,17 +29,17 @@ import lombok.Setter; public class DevConfigEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 配置键 */ - @Schema(description = "配置键", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "配置键") @NotBlank(message = "configKey不能为空") private String configKey; /** 配置值 */ - @Schema(description = "配置值", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "配置值") @NotBlank(message = "configValue不能为空") private String configValue; @@ -48,7 +48,7 @@ public class DevConfigEditParam { private String remark; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigIdParam.java index 228d6a28..64ec8716 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevConfigIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorOrgListParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorOrgListParam.java new file mode 100644 index 00000000..e18dbcac --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorOrgListParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.config.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; + +/** + * 组织列表选择器参数 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +public class DevConfigSelectorOrgListParam { + + /** 当前页 */ + @Schema(description = "当前页码") + private Integer current; + + /** 每页条数 */ + @Schema(description = "每页条数") + private Integer size; + + /** 父id */ + @Schema(description = "父id") + private String parentId; + + /** 名称关键词 */ + @Schema(description = "名称关键词") + private String searchKey; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorPositionParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorPositionParam.java new file mode 100644 index 00000000..79879f3f --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorPositionParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.config.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; + +/** + * 职位选择器参数 + * + * @author xuyuxiang + * @date 2022/7/26 15:58 + **/ +@Getter +@Setter +public class DevConfigSelectorPositionParam { + + /** 当前页 */ + @Schema(description = "当前页码") + private Integer current; + + /** 每页条数 */ + @Schema(description = "每页条数") + private Integer size; + + /** 组织id */ + @Schema(description = "组织id") + private String orgId; + + /** 名称关键词 */ + @Schema(description = "名称关键词") + private String searchKey; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorRoleParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorRoleParam.java new file mode 100644 index 00000000..6d7af091 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/param/DevConfigSelectorRoleParam.java @@ -0,0 +1,48 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.config.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; + +/** + * 角色选择器参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:02 + **/ +@Getter +@Setter +public class DevConfigSelectorRoleParam { + + /** 当前页 */ + @Schema(description = "当前页码") + private Integer current; + + /** 每页条数 */ + @Schema(description = "每页条数") + private Integer size; + + /** 组织id */ + @Schema(description = "组织id") + private String orgId; + + /** 角色分类 */ + @Schema(description = "角色分类") + private String category; + + /** 名称关键词 */ + @Schema(description = "名称关键词") + private String searchKey; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/service/impl/DevConfigServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/service/impl/DevConfigServiceImpl.java index c5a256e8..9832d019 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/service/impl/DevConfigServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/config/service/impl/DevConfigServiceImpl.java @@ -50,8 +50,6 @@ public class DevConfigServiceImpl extends ServiceImpl sysBaseList() { DevConfigListParam devConfigListParam = new DevConfigListParam(); devConfigListParam.setCategory(DevConfigCategoryEnum.SYS_BASE.getValue()); - return this.list(devConfigListParam).stream().filter(devConfig -> !devConfig.getConfigKey() - .equals(SNOWY_SYS_DEFAULT_PASSWORD_KEY)).collect(Collectors.toList()); + return this.list(devConfigListParam); } @Override @@ -104,7 +101,7 @@ public class DevConfigServiceImpl extends ServiceImpl lambdaQueryWrapper = new LambdaQueryWrapper<>(); // 查询部分字段 lambdaQueryWrapper.select(DevConfig::getId, DevConfig::getConfigKey, DevConfig::getConfigValue, - DevConfig::getCategory, DevConfig::getSortCode); + DevConfig::getCategory, DevConfig::getSortCode, DevConfig::getRemark); if(ObjectUtil.isNotEmpty(devConfigListParam.getCategory())) { lambdaQueryWrapper.eq(DevConfig::getCategory, devConfigListParam.getCategory()); } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/controller/DevDictController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/controller/DevDictController.java index 6b7b990f..5ba8bac7 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/controller/DevDictController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/controller/DevDictController.java @@ -14,10 +14,11 @@ package vip.xiaonuo.dev.modular.dict.controller; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -30,6 +31,7 @@ import vip.xiaonuo.dev.modular.dict.entity.DevDict; import vip.xiaonuo.dev.modular.dict.param.*; import vip.xiaonuo.dev.modular.dict.service.DevDictService; +import javax.validation.Valid; import java.util.List; /** @@ -39,6 +41,7 @@ import java.util.List; * @date 2022/6/21 14:58 **/ @Tag(name = "字典控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 2) @RestController @Validated public class DevDictController { @@ -52,6 +55,7 @@ public class DevDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取字典分页") @GetMapping("/dev/dict/page") public CommonResult> page(DevDictPageParam devDictPageParam) { @@ -64,6 +68,7 @@ public class DevDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取字典列表") @GetMapping("/dev/dict/list") public CommonResult> list(DevDictListParam devDictListParam) { @@ -76,6 +81,7 @@ public class DevDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 3) @Operation(summary = "获取字典树") @GetMapping("/dev/dict/tree") public CommonResult>> tree(DevDictTreeParam devDictTreeParam) { @@ -88,6 +94,7 @@ public class DevDictController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "添加字典") @CommonLog("添加字典") @PostMapping("/dev/dict/add") @@ -102,6 +109,7 @@ public class DevDictController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 5) @Operation(summary = "编辑字典") @CommonLog("编辑字典") @PostMapping("/dev/dict/edit") @@ -116,6 +124,7 @@ public class DevDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "删除字典") @CommonLog("删除字典") @PostMapping("/dev/dict/delete") @@ -131,6 +140,7 @@ public class DevDictController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取字典详情") @GetMapping("/dev/dict/detail") public CommonResult detail(@Valid DevDictIdParam devDictIdParam) { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/entity/DevDict.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/entity/DevDict.java index ff44cda1..6b795abf 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/entity/DevDict.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/entity/DevDict.java @@ -47,6 +47,10 @@ public class DevDict extends CommonEntity { @Schema(description = "字典值") private String dictValue; + /** 编码 */ + @Schema(description = "编码") + private String code; + /** 分类 */ @Schema(description = "分类") private String category; @@ -57,6 +61,6 @@ public class DevDict extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/mapper/mapping/DevDictMapper.xml b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/mapper/mapping/DevDictMapper.xml index 732eb08b..4e23125c 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/mapper/mapping/DevDictMapper.xml +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/mapper/mapping/DevDictMapper.xml @@ -2,5 +2,4 @@ - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/param/DevDictIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/param/DevDictIdParam.java index 5d420c2d..a8270d24 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/param/DevDictIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/param/DevDictIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevDictIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/service/impl/DevDictServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/service/impl/DevDictServiceImpl.java index 403a0682..50a773fb 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/service/impl/DevDictServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/dict/service/impl/DevDictServiceImpl.java @@ -20,6 +20,7 @@ import cn.hutool.core.lang.tree.TreeNodeConfig; import cn.hutool.core.lang.tree.TreeUtil; import cn.hutool.core.lang.tree.parser.DefaultNodeParser; import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.RandomUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.json.JSONUtil; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; @@ -30,6 +31,7 @@ import com.fhs.trans.service.impl.DictionaryTransService; import jakarta.annotation.Resource; import org.springframework.beans.factory.InitializingBean; import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.page.CommonPageRequest; @@ -108,8 +110,8 @@ public class DevDictServiceImpl extends ServiceImpl impl } List devDictList = this.list(lambdaQueryWrapper); List> treeNodeList = devDictList.stream().map(devDict -> - new TreeNode<>(devDict.getId(), devDict.getParentId(), - devDict.getDictLabel(), devDict.getSortCode()).setExtra(JSONUtil.parseObj(devDict))) + new TreeNode<>(devDict.getId(), devDict.getParentId(), + devDict.getDictLabel(), devDict.getSortCode()).setExtra(JSONUtil.parseObj(devDict))) .collect(Collectors.toList()); // 精简冗余字段(sortCode、weight字段合并) TreeNodeConfig treeNodeConfig = new TreeNodeConfig(); @@ -121,6 +123,7 @@ public class DevDictServiceImpl extends ServiceImpl impl public void add(DevDictAddParam devDictAddParam) { checkParam(devDictAddParam); DevDict devDict = BeanUtil.toBean(devDictAddParam, DevDict.class); + devDict.setCode(RandomUtil.randomString(10)); this.save(devDict); refreshTransCache(); } @@ -172,6 +175,7 @@ public class DevDictServiceImpl extends ServiceImpl impl } } + @Transactional(rollbackFor = Exception.class) @Override public void delete(List devDictIdParamList) { List devDictIdList = CollStreamUtil.toList(devDictIdParamList, DevDictIdParam::getId); @@ -205,6 +209,12 @@ public class DevDictServiceImpl extends ServiceImpl impl refreshTransCache(); } + /** + * 刷新字典缓存 + * + * @author xuyuxiang + * @date 2022/9/26 15:33 + **/ private void refreshTransCache() { // 异步不阻塞主线程,不会 增加启动用时 CompletableFuture.supplyAsync(() -> { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/controller/DevEmailController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/controller/DevEmailController.java index 78a24eca..f12a618b 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/controller/DevEmailController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/controller/DevEmailController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.dev.modular.email.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -29,6 +30,7 @@ import vip.xiaonuo.dev.modular.email.entity.DevEmail; import vip.xiaonuo.dev.modular.email.param.*; import vip.xiaonuo.dev.modular.email.service.DevEmailService; +import javax.validation.Valid; import java.util.List; /** @@ -38,6 +40,7 @@ import java.util.List; * @date 2022/2/23 18:26 **/ @Tag(name = "邮件控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 3) @RestController @Validated public class DevEmailController { @@ -45,12 +48,43 @@ public class DevEmailController { @Resource private DevEmailService devEmailService; + /** + * 动态发送TXT邮件 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 1) + @Operation(summary = "动态发送TXT邮件") + @CommonLog("动态发送TXT邮件") + @PostMapping("/dev/email/sendDynamicTxt") + public CommonResult sendDynamic(@RequestBody @Valid DevEmailSendDynamicTxtParam devEmailSendDynamicTxtParam) { + devEmailService.sendDynamicTxt(devEmailSendDynamicTxtParam); + return CommonResult.ok(); + } + + /** + * 动态发送HTML邮件 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 2) + @Operation(summary = "动态发送HTML邮件") + @CommonLog("动态发送HTML邮件") + @PostMapping("/dev/email/sendDynamicHtml") + public CommonResult sendDynamic(@RequestBody @Valid DevEmailSendDynamicHtmlParam devEmailSendDynamicHtmlParam) { + devEmailService.sendDynamicHtml(devEmailSendDynamicHtmlParam); + return CommonResult.ok(); + } + /** * 发送邮件——本地TXT * * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "发送本地文本邮件") @CommonLog("发送本地文本邮件") @PostMapping("/dev/email/sendLocalTxt") @@ -65,6 +99,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "发送本地HTML邮件") @CommonLog("发送本地HTML邮件") @PostMapping("/dev/email/sendLocalHtml") @@ -79,6 +114,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 5) @Operation(summary = "发送阿里云文本邮件") @CommonLog("发送阿里云文本邮件") @PostMapping("/dev/email/sendAliyunTxt") @@ -93,6 +129,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 6) @Operation(summary = "发送阿里云HTML邮件") @CommonLog("发送阿里云HTML邮件") @PostMapping("/dev/email/sendAliyunHtml") @@ -107,6 +144,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 7) @Operation(summary = "发送阿里云模板邮件") @CommonLog("发送阿里云模板邮件") @PostMapping("/dev/email/sendAliyunTmp") @@ -121,6 +159,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 8) @Operation(summary = "发送腾讯云文本邮件") @CommonLog("发送腾讯云文本邮件") @PostMapping("/dev/email/sendTencentTxt") @@ -135,6 +174,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 9) @Operation(summary = "发送腾讯云HTML邮件") @CommonLog("发送腾讯云HTML邮件") @PostMapping("/dev/email/sentTencentHtml") @@ -149,6 +189,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 10) @Operation(summary = "发送腾讯云模板邮件") @CommonLog("发送腾讯云模板邮件") @PostMapping("/dev/email/sentTencentTmp") @@ -163,6 +204,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 11) @Operation(summary = "获取邮件分页") @GetMapping("/dev/email/page") public CommonResult> page(DevEmailPageParam devEmailPageParam) { @@ -175,11 +217,12 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 12) @Operation(summary = "删除邮件") @CommonLog("删除邮件") @PostMapping("/dev/email/delete") public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List devEmailIdParamList) { + List devEmailIdParamList) { devEmailService.delete(devEmailIdParamList); return CommonResult.ok(); } @@ -190,6 +233,7 @@ public class DevEmailController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 13) @Operation(summary = "获取邮件详情") @GetMapping("/dev/email/detail") public CommonResult detail(@Valid DevEmailIdParam devEmailIdParam) { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/entity/DevEmail.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/entity/DevEmail.java index 00dd7642..16e4fcc7 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/entity/DevEmail.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/entity/DevEmail.java @@ -30,7 +30,7 @@ import vip.xiaonuo.common.pojo.CommonEntity; public class DevEmail extends CommonEntity { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 邮件引擎 */ diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailIdParam.java index 46585116..67ba872b 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevEmailIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunHtmlParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunHtmlParam.java index 8ced5a3a..a4ab32fc 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunHtmlParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunHtmlParam.java @@ -28,22 +28,22 @@ import lombok.Setter; public class DevEmailSendAliyunHtmlParam { /** 发件人邮箱 */ - @Schema(description = "发件人邮箱,即管理控制台中配置的发信地址", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "发件人邮箱,即管理控制台中配置的发信地址") @NotBlank(message = "sendAccount不能为空") private String sendAccount; /** 接收人 */ - @Schema(description = "接收人邮箱地址,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "接收人邮箱地址,多个逗号拼接") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 邮件主题 */ - @Schema(description = "邮件主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件主题") @NotBlank(message = "subject不能为空") private String subject; /** 邮件正文 */ - @Schema(description = "邮件正文", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件正文") @NotBlank(message = "content不能为空") private String content; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTmpParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTmpParam.java index 5d5b4a5c..47fd9b69 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTmpParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTmpParam.java @@ -28,17 +28,17 @@ import lombok.Setter; public class DevEmailSendAliyunTmpParam { /** 发件人邮箱 */ - @Schema(description = "管理控制台中配置的发信地址", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "管理控制台中配置的发信地址") @NotBlank(message = "sendAccount不能为空") private String sendAccount; /** 接收人 */ - @Schema(description = "预先创建且上传了收件人的收件人列表名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "预先创建且上传了收件人的收件人列表名称") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 模板名 */ - @Schema(description = "预先创建且通过审核的模板名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "预先创建且通过审核的模板名称") @NotBlank(message = "templateName不能为空") private String templateName; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTxtParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTxtParam.java index a6005fcb..91025436 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTxtParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendAliyunTxtParam.java @@ -28,22 +28,22 @@ import lombok.Setter; public class DevEmailSendAliyunTxtParam { /** 发件人邮箱 */ - @Schema(description = "发件人邮箱,即管理控制台中配置的发信地址", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "发件人邮箱,即管理控制台中配置的发信地址") @NotBlank(message = "sendAccount不能为空") private String sendAccount; /** 接收人 */ - @Schema(description = "接收人邮箱地址,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "接收人邮箱地址,多个逗号拼接") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 邮件主题 */ - @Schema(description = "邮件主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件主题") @NotBlank(message = "subject不能为空") private String subject; /** 邮件正文 */ - @Schema(description = "邮件正文", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件正文") @NotBlank(message = "content不能为空") private String content; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendDynamicHtmlParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendDynamicHtmlParam.java new file mode 100644 index 00000000..02136793 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendDynamicHtmlParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.email.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 邮件发送——本地HTML参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevEmailSendDynamicHtmlParam { + + /** 接收人 */ + @Schema(description = "接收人邮箱地址,多个逗号拼接") + @NotBlank(message = "receiveAccounts不能为空") + private String receiveAccounts; + + /** 邮件主题 */ + @Schema(description = "邮件主题") + @NotBlank(message = "subject不能为空") + private String subject; + + /** 邮件正文 */ + @Schema(description = "邮件正文") + @NotBlank(message = "content不能为空") + private String content; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendDynamicTxtParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendDynamicTxtParam.java new file mode 100644 index 00000000..b5aec40f --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendDynamicTxtParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.email.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 邮件发送——本地HTML参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevEmailSendDynamicTxtParam { + + /** 接收人 */ + @Schema(description = "接收人邮箱地址,多个逗号拼接") + @NotBlank(message = "receiveAccounts不能为空") + private String receiveAccounts; + + /** 邮件主题 */ + @Schema(description = "邮件主题") + @NotBlank(message = "subject不能为空") + private String subject; + + /** 邮件正文 */ + @Schema(description = "邮件正文") + @NotBlank(message = "content不能为空") + private String content; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalHtmlParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalHtmlParam.java index 32554513..8db478d0 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalHtmlParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalHtmlParam.java @@ -35,17 +35,17 @@ import java.util.Map; public class DevEmailSendLocalHtmlParam { /** 接收人 */ - @Schema(description = "接收人邮箱地址,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "接收人邮箱地址,多个逗号拼接") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 邮件主题 */ - @Schema(description = "邮件主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件主题") @NotBlank(message = "subject不能为空") private String subject; /** 邮件正文 */ - @Schema(description = "邮件正文", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件正文") @NotBlank(message = "content不能为空") private String content; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalTxtParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalTxtParam.java index 145641b4..3947b191 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalTxtParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendLocalTxtParam.java @@ -32,17 +32,17 @@ import java.util.List; public class DevEmailSendLocalTxtParam { /** 接收人 */ - @Schema(description = "接收人邮箱地址,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "接收人邮箱地址,多个逗号拼接") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 邮件主题 */ - @Schema(description = "邮件主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件主题") @NotBlank(message = "subject不能为空") private String subject; /** 邮件正文 */ - @Schema(description = "邮件正文", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件正文") @NotBlank(message = "content不能为空") private String content; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentHtmlParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentHtmlParam.java index a5bd06e4..0f7db9d5 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentHtmlParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentHtmlParam.java @@ -32,22 +32,22 @@ import java.util.List; public class DevEmailSendTencentHtmlParam { /** 发件人邮箱 */ - @Schema(description = "管理控制台中配置的发信地址", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "管理控制台中配置的发信地址") @NotBlank(message = "sendAccount不能为空") private String sendAccount; /** 接收人 */ - @Schema(description = "接收人邮箱地址,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "接收人邮箱地址,多个逗号拼接") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 邮件主题 */ - @Schema(description = "邮件主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件主题") @NotBlank(message = "subject不能为空") private String subject; /** 邮件正文 */ - @Schema(description = "邮件正文", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件正文") @NotBlank(message = "content不能为空") private String content; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTmpParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTmpParam.java index 1f1d2dae..7e64a9ae 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTmpParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTmpParam.java @@ -32,22 +32,22 @@ import java.util.List; public class DevEmailSendTencentTmpParam { /** 发件人邮箱 */ - @Schema(description = "管理控制台中配置的发信地址", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "管理控制台中配置的发信地址") @NotBlank(message = "sendAccount不能为空") private String sendAccount; /** 接收人 */ - @Schema(description = "预先创建且上传了收件人的收件人列表id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "预先创建且上传了收件人的收件人列表id") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 模板名 */ - @Schema(description = "预先创建且通过审核的模板Id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "预先创建且通过审核的模板Id") @NotBlank(message = "templateName不能为空") private String templateName; /** 邮件主题 */ - @Schema(description = "邮件主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件主题") @NotBlank(message = "subject不能为空") private String subject; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTxtParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTxtParam.java index beeeb2c2..bb8ee72f 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTxtParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/param/DevEmailSendTencentTxtParam.java @@ -32,22 +32,22 @@ import java.util.List; public class DevEmailSendTencentTxtParam { /** 发件人邮箱 */ - @Schema(description = "管理控制台中配置的发信地址", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "管理控制台中配置的发信地址") @NotBlank(message = "sendAccount不能为空") private String sendAccount; /** 接收人 */ - @Schema(description = "接收人邮箱地址,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "接收人邮箱地址,多个逗号拼接") @NotBlank(message = "receiveAccounts不能为空") private String receiveAccounts; /** 邮件主题 */ - @Schema(description = "邮件主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件主题") @NotBlank(message = "subject不能为空") private String subject; /** 邮件正文 */ - @Schema(description = "邮件正文", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮件正文") @NotBlank(message = "content不能为空") private String content; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/provider/DevEmailApiProvider.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/provider/DevEmailApiProvider.java index a3829fd1..e8790334 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/provider/DevEmailApiProvider.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/provider/DevEmailApiProvider.java @@ -36,6 +36,24 @@ public class DevEmailApiProvider implements DevEmailApi { @Resource private DevEmailService devEmailService; + @Override + public void sendDynamicTxtEmail(String tos, String subject, String content) { + DevEmailSendDynamicTxtParam sendDynamicTxtParam = new DevEmailSendDynamicTxtParam(); + sendDynamicTxtParam.setReceiveAccounts(tos); + sendDynamicTxtParam.setSubject(subject); + sendDynamicTxtParam.setContent(content); + devEmailService.sendDynamicTxt(sendDynamicTxtParam); + } + + @Override + public void sendDynamicHtmlEmail(String tos, String subject, String content) { + DevEmailSendDynamicHtmlParam sendDynamicHtmlParam = new DevEmailSendDynamicHtmlParam(); + sendDynamicHtmlParam.setReceiveAccounts(tos); + sendDynamicHtmlParam.setSubject(subject); + sendDynamicHtmlParam.setContent(content); + devEmailService.sendDynamicHtml(sendDynamicHtmlParam); + } + @Override public void sendTextEmailLocal(String tos, String subject, String content, List files) { DevEmailSendLocalTxtParam devEmailSendLocalTxtParam = new DevEmailSendLocalTxtParam(); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/DevEmailService.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/DevEmailService.java index cb03154f..146f6949 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/DevEmailService.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/DevEmailService.java @@ -27,6 +27,46 @@ import java.util.List; **/ public interface DevEmailService extends IService { + /** + * 动态发送TXT邮件 + * + * @param engine 发送引擎 + * @param receiveAccounts 收件人邮箱,逗号拼接 + * @param subject 邮件主题 + * @param content 邮件内容 + * @author xuyuxiang + * @date 2022/2/7 22:29 + */ + void sendDynamicTxt(String engine, String receiveAccounts, String subject, String content); + + /** + * 动态发送HTML邮件 + * + * @param engine 发送引擎 + * @param receiveAccounts 收件人邮箱,逗号拼接 + * @param subject 邮件主题 + * @param content 邮件内容 + * @author xuyuxiang + * @date 2022/2/7 22:29 + */ + void sendDynamicHtml(String engine, String receiveAccounts, String subject, String content); + + /** + * 动态发送TXT邮件 + * + * @author xuyuxiang + * @date 2022/6/21 18:37 + **/ + void sendDynamicTxt(DevEmailSendDynamicTxtParam devEmailSendDynamicTxtParam); + + /** + * 动态发送HTML邮件 + * + * @author xuyuxiang + * @date 2022/6/21 18:37 + **/ + void sendDynamicHtml(DevEmailSendDynamicHtmlParam devEmailSendDynamicHtmlParam); + /** * 发送邮件——本地TXT * diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/impl/DevEmailServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/impl/DevEmailServiceImpl.java index 674e0ecd..71c158cf 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/impl/DevEmailServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/service/impl/DevEmailServiceImpl.java @@ -21,12 +21,14 @@ import cn.hutool.extra.mail.MailAccount; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.page.CommonPageRequest; import vip.xiaonuo.common.util.CommonEmailUtil; +import vip.xiaonuo.dev.api.DevConfigApi; import vip.xiaonuo.dev.modular.email.entity.DevEmail; import vip.xiaonuo.dev.modular.email.enums.DevEmailEngineTypeEnum; import vip.xiaonuo.dev.modular.email.mapper.DevEmailMapper; @@ -47,6 +49,90 @@ import java.util.List; @Service public class DevEmailServiceImpl extends ServiceImpl implements DevEmailService { + /** 默认邮件引擎 */ + private static final String SNOWY_SYS_DEFAULT_EMAIL_ENGINE_KEY = "SNOWY_SYS_DEFAULT_EMAIL_ENGINE"; + + @Resource + private DevConfigApi devConfigApi; + + @Transactional(rollbackFor = Exception.class) + @Override + public void sendDynamicTxt(String engine, String receiveAccounts, String subject, String content) { + if(engine.equals(DevEmailEngineTypeEnum.LOCAL.getValue())) { + DevEmailSendLocalTxtParam devEmailSendLocalTxtParam = new DevEmailSendLocalTxtParam(); + devEmailSendLocalTxtParam.setReceiveAccounts(receiveAccounts); + devEmailSendLocalTxtParam.setSubject(subject); + devEmailSendLocalTxtParam.setContent(content); + this.sendLocal(devEmailSendLocalTxtParam); + } else if (engine.equals(DevEmailEngineTypeEnum.ALIYUN.getValue())) { + DevEmailSendAliyunTxtParam devEmailSendAliyunTxtParam = new DevEmailSendAliyunTxtParam(); + devEmailSendAliyunTxtParam.setReceiveAccounts(receiveAccounts); + devEmailSendAliyunTxtParam.setSubject(subject); + devEmailSendAliyunTxtParam.setContent(content); + this.sendAliyun(devEmailSendAliyunTxtParam); + } else if (engine.equals(DevEmailEngineTypeEnum.TENCENT.getValue())) { + DevEmailSendTencentTxtParam devEmailSendTencentTxtParam = new DevEmailSendTencentTxtParam(); + devEmailSendTencentTxtParam.setReceiveAccounts(receiveAccounts); + devEmailSendTencentTxtParam.setSubject(subject); + devEmailSendTencentTxtParam.setContent(content); + this.sendTencent(devEmailSendTencentTxtParam); + } else { + throw new CommonException("不支持的邮件引擎:{}", engine); + } + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void sendDynamicHtml(String engine, String receiveAccounts, String subject, String content) { + if(engine.equals(DevEmailEngineTypeEnum.LOCAL.getValue())) { + DevEmailSendLocalHtmlParam devEmailSendLocalHtmlParam = new DevEmailSendLocalHtmlParam(); + devEmailSendLocalHtmlParam.setReceiveAccounts(receiveAccounts); + devEmailSendLocalHtmlParam.setSubject(subject); + devEmailSendLocalHtmlParam.setContent(content); + this.sendLocal(devEmailSendLocalHtmlParam); + } else if (engine.equals(DevEmailEngineTypeEnum.ALIYUN.getValue())) { + DevEmailSendAliyunHtmlParam devEmailSendAliyunHtmlParam = new DevEmailSendAliyunHtmlParam(); + devEmailSendAliyunHtmlParam.setReceiveAccounts(receiveAccounts); + devEmailSendAliyunHtmlParam.setSubject(subject); + devEmailSendAliyunHtmlParam.setContent(content); + this.sendAliyun(devEmailSendAliyunHtmlParam); + } else if (engine.equals(DevEmailEngineTypeEnum.TENCENT.getValue())) { + DevEmailSendTencentHtmlParam devEmailSendTencentHtmlParam = new DevEmailSendTencentHtmlParam(); + devEmailSendTencentHtmlParam.setReceiveAccounts(receiveAccounts); + devEmailSendTencentHtmlParam.setSubject(subject); + devEmailSendTencentHtmlParam.setContent(content); + this.sendTencent(devEmailSendTencentHtmlParam); + } else { + throw new CommonException("不支持的邮件引擎:{}", engine); + } + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void sendDynamicTxt(DevEmailSendDynamicTxtParam devEmailSendDynamicTxtParam) { + String defaultEmailEngine = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_EMAIL_ENGINE_KEY); + if(ObjectUtil.isEmpty(defaultEmailEngine)) { + throw new CommonException("请联系管理员配置默认邮件发送引擎"); + } + String receiveAccounts = devEmailSendDynamicTxtParam.getReceiveAccounts(); + String subject = devEmailSendDynamicTxtParam.getSubject(); + String content = devEmailSendDynamicTxtParam.getContent(); + this.sendDynamicTxt(defaultEmailEngine, receiveAccounts, subject, content); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void sendDynamicHtml(DevEmailSendDynamicHtmlParam devEmailSendDynamicHtmlParam) { + String defaultEmailEngine = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_EMAIL_ENGINE_KEY); + if(ObjectUtil.isEmpty(defaultEmailEngine)) { + throw new CommonException("请联系管理员配置默认邮件发送引擎"); + } + String receiveAccounts = devEmailSendDynamicHtmlParam.getReceiveAccounts(); + String subject = devEmailSendDynamicHtmlParam.getSubject(); + String content = devEmailSendDynamicHtmlParam.getContent(); + this.sendDynamicHtml(defaultEmailEngine, receiveAccounts, subject, content); + } + @Transactional(rollbackFor = Exception.class) @Override public void sendLocal(DevEmailSendLocalTxtParam devEmailSendLocalTxtParam) { @@ -180,6 +266,7 @@ public class DevEmailServiceImpl extends ServiceImpl i return this.page(CommonPageRequest.defaultPage(), queryWrapper); } + @Transactional(rollbackFor = Exception.class) @Override public void delete(List devEmailIdParamList) { this.removeByIds(CollStreamUtil.toList(devEmailIdParamList, DevEmailIdParam::getId)); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailAliyunUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailAliyunUtil.java index 8b2f0090..9c195940 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailAliyunUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailAliyunUtil.java @@ -24,7 +24,7 @@ import vip.xiaonuo.dev.api.DevConfigApi; /** * 阿里云邮件工具类 - * 参考文档:https://help.aliyun.com/document_detail/29459.html + * 参考文档:参考文档 * * @author xuyuxiang * @date 2022/6/17 10:17 @@ -36,7 +36,7 @@ public class DevEmailAliyunUtil { private static final String SNOWY_EMAIL_ALIYUN_ACCESS_KEY_ID_KEY = "SNOWY_EMAIL_ALIYUN_ACCESS_KEY_ID"; private static final String SNOWY_EMAIL_ALIYUN_ACCESS_KEY_SECRET_KEY = "SNOWY_EMAIL_ALIYUN_ACCESS_KEY_SECRET"; - private static final String SNOWY_EMAIL_ALIYUN_REGION_ID_KEY = "SNOWY_EMAIL_ALIYUN_REGION_ID"; + private static final String SNOWY_EMAIL_ALIYUN_FROM_KEY = "SNOWY_EMAIL_ALIYUN_FROM"; /** * 初始化操作的客户端 @@ -61,16 +61,8 @@ public class DevEmailAliyunUtil { if(ObjectUtil.isEmpty(accessKeySecret)) { throw new CommonException("阿里云邮件操作客户端未正确配置:accessKeySecret为空"); } - - /* regionId */ - String regionId = devConfigApi.getValueByKey(SNOWY_EMAIL_ALIYUN_REGION_ID_KEY); - - if(ObjectUtil.isEmpty(regionId)) { - throw new CommonException("阿里云邮件操作客户端未正确配置:regionId为空"); - } - try { - client = new Client(new Config().setRegionId(regionId).setEndpoint("dm.aliyuncs.com").setAccessKeyId(accessKeyId).setAccessKeySecret(accessKeySecret)); + client = new Client(new Config().setRegionId("cn-hangzhou").setEndpoint("dm.aliyuncs.com").setAccessKeyId(accessKeyId).setAccessKeySecret(accessKeySecret)); } catch (Exception e) { throw new CommonException(e.getMessage()); } @@ -91,6 +83,9 @@ public class DevEmailAliyunUtil { public static String sendTextEmail(String from, String user, String tos, String subject, String content) { try { initClient(); + if(ObjectUtil.isEmpty(from)) { + from = getDefaultFrom(); + } SingleSendMailRequest singleSendMailRequest = createSingleSendRequest(from, user, tos, subject, content, false); return client.singleSendMail(singleSendMailRequest).getBody().getEnvId(); } catch (Exception e) { @@ -113,6 +108,9 @@ public class DevEmailAliyunUtil { public static String sendHtmlEmail(String from, String user, String tos, String subject, String content) { try { initClient(); + if(ObjectUtil.isEmpty(from)) { + from = getDefaultFrom(); + } SingleSendMailRequest singleSendMailRequest = createSingleSendRequest(from, user, tos, subject, content, true); return client.singleSendMail(singleSendMailRequest).getBody().getEnvId(); } catch (Exception e) { @@ -134,6 +132,9 @@ public class DevEmailAliyunUtil { public static String sendEmailWithTemplate(String from, String tagName, String toName, String templateName) { try { initClient(); + if(ObjectUtil.isEmpty(from)) { + from = getDefaultFrom(); + } BatchSendMailRequest batchSendMailRequest = createBatchSendRequest(from, tagName, toName, templateName); return client.batchSendMail(batchSendMailRequest).getBody().getEnvId(); } catch (Exception e) { @@ -211,4 +212,19 @@ public class DevEmailAliyunUtil { return request; } + + /** + * 获取默认发送账号 + * + * @author xuyuxiang + * @date 2024/1/26 16:40 + **/ + public static String getDefaultFrom() { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + String signName = devConfigApi.getValueByKey(SNOWY_EMAIL_ALIYUN_FROM_KEY); + if(ObjectUtil.isEmpty(signName)) { + throw new CommonException("阿里云邮件操作客户端未正确配置:from为空"); + } + return signName; + } } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailLocalUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailLocalUtil.java index 897f07aa..fd480c18 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailLocalUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailLocalUtil.java @@ -39,6 +39,7 @@ public class DevEmailLocalUtil { private static MailAccount mailAccount; private static final String SNOWY_EMAIL_LOCAL_FROM_KEY = "SNOWY_EMAIL_LOCAL_FROM"; + private static final String SNOWY_EMAIL_LOCAL_PASSWORD_KEY = "SNOWY_EMAIL_LOCAL_PASSWORD"; private static final String SNOWY_EMAIL_LOCAL_SMTP_HOST_KEY = "SNOWY_EMAIL_LOCAL_SMTP_HOST"; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailTencentUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailTencentUtil.java index 417ac19b..656aeef2 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailTencentUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/email/util/DevEmailTencentUtil.java @@ -31,7 +31,7 @@ import java.util.List; /** * 腾讯云邮件工具类 - * 参考文档:https://cloud.tencent.com/document/api/1288/51034 + * 参考文档:参考文档 * * @author xuyuxiang * @date 2022/6/17 11:26 @@ -43,7 +43,7 @@ public class DevEmailTencentUtil { private static final String SNOWY_EMAIL_TENCENT_SECRET_ID_KEY = "SNOWY_EMAIL_TENCENT_SECRET_ID"; private static final String SNOWY_EMAIL_TENCENT_SECRET_KEY_KEY = "SNOWY_EMAIL_TENCENT_SECRET_KEY"; - private static final String SNOWY_EMAIL_TENCENT_REGION_ID_KEY = "SNOWY_EMAIL_TENCENT_REGION_ID"; + private static final String SNOWY_EMAIL_TENCENT_FROM_KEY = "SNOWY_EMAIL_TENCENT_FROM"; /** * 初始化操作的客户端 @@ -68,15 +68,7 @@ public class DevEmailTencentUtil { if(ObjectUtil.isEmpty(secretKey)) { throw new CommonException("腾讯云邮件操作客户端未正确配置:secretKey为空"); } - - /* regionId */ - String regionId = devConfigApi.getValueByKey(SNOWY_EMAIL_TENCENT_REGION_ID_KEY); - - if(ObjectUtil.isEmpty(regionId)) { - throw new CommonException("腾讯云邮件操作客户端未正确配置:regionId为空"); - } - - client = new SesClient(new Credential(secretId, secretKey), regionId); + client = new SesClient(new Credential(secretId, secretKey), "ap-guangzhou"); } /** @@ -88,7 +80,7 @@ public class DevEmailTencentUtil { * @param subject 邮件主题,必传 * @param content 邮件 txt 正文,必传,注意:腾讯云api目前要求请求包大小不得超过8 MB。 * @param attachmentList 需要发送附件时,填写附件相关参数,格式:[{"FileName": "xxxx", "Content": "xxx"}] - * 支持的格式与说明见:https://cloud.tencent.com/document/api/1288/51053#Attachment + * 支持的格式与说明见:参考文档 * @return 发送成功的回执id * @author xuyuxiang * @date 2022/2/23 14:24 @@ -96,6 +88,9 @@ public class DevEmailTencentUtil { public static String sendTextEmail(String from, String user, String tos, String subject, String content, List attachmentList) { try { initClient(); + if(ObjectUtil.isEmpty(from)) { + from = getDefaultFrom(); + } SendEmailRequest singleSendMailRequest = createSingleSendRequest(from, user, tos, subject, content, false, attachmentList); return client.SendEmail(singleSendMailRequest).getMessageId(); } catch (TencentCloudSDKException e) { @@ -112,7 +107,7 @@ public class DevEmailTencentUtil { * @param subject 邮件主题,必传 * @param content 邮件 txt 正文,必传,注意:腾讯云api目前要求请求包大小不得超过8 MB。 * @param attachmentList 需要发送附件时,填写附件相关参数,格式:[{"FileName": "xxxx", "Content": "xxx"}] - * 支持的格式与说明见:https://cloud.tencent.com/document/api/1288/51053#Attachment + * 支持的格式与说明见:参考文档 * @return 发送成功的回执id * @author xuyuxiang * @date 2022/2/23 14:24 @@ -120,6 +115,9 @@ public class DevEmailTencentUtil { public static String sendHtmlEmail(String from, String user, String tos, String subject, String content, List attachmentList) { try { initClient(); + if(ObjectUtil.isEmpty(from)) { + from = getDefaultFrom(); + } SendEmailRequest singleSendMailRequest = createSingleSendRequest(from, user, tos, subject, content, true, attachmentList); return client.SendEmail(singleSendMailRequest).getMessageId(); } catch (TencentCloudSDKException e) { @@ -137,7 +135,7 @@ public class DevEmailTencentUtil { * @param templateParam 预先创建且通过审核的模板的参数json,格式{"name":"张三"},可不传 * @param subject 邮件主题,必传 * @param attachmentList 需要发送附件时,填写附件相关参数,格式:[{"FileName": "xxxx", "Content": "xxx"}] - * 支持的格式与说明见:https://cloud.tencent.com/document/api/1288/51053#Attachment + * 支持的格式与说明见:参考文档 * @return 发送成功的回执id * @author xuyuxiang * @date 2022/2/23 14:24 @@ -146,6 +144,9 @@ public class DevEmailTencentUtil { String templateParam, String subject, List attachmentList) { try { initClient(); + if(ObjectUtil.isEmpty(from)) { + from = getDefaultFrom(); + } BatchSendEmailRequest batchSendEmailRequest = createBatchSendRequest(from, user, toId, templateId, templateParam, subject, attachmentList); return client.BatchSendEmail(batchSendEmailRequest).getTaskId().toString(); } catch (TencentCloudSDKException e) { @@ -210,4 +211,19 @@ public class DevEmailTencentUtil { } return batchSendEmailRequest; } + + /** + * 获取默认发送账号 + * + * @author xuyuxiang + * @date 2024/1/26 16:40 + **/ + public static String getDefaultFrom() { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + String signName = devConfigApi.getValueByKey(SNOWY_EMAIL_TENCENT_FROM_KEY); + if(ObjectUtil.isEmpty(signName)) { + throw new CommonException("腾讯云邮件操作客户端未正确配置:from为空"); + } + return signName; + } } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/controller/DevFileController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/controller/DevFileController.java index 240c58ea..f4d81e1c 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/controller/DevFileController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/controller/DevFileController.java @@ -13,11 +13,12 @@ package vip.xiaonuo.dev.modular.file.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; import jakarta.servlet.http.HttpServletResponse; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.http.MediaType; import org.springframework.validation.annotation.Validated; @@ -34,6 +35,7 @@ import vip.xiaonuo.dev.modular.file.param.DevFilePageParam; import vip.xiaonuo.dev.modular.file.param.DevFileUrlListParam; import vip.xiaonuo.dev.modular.file.service.DevFileService; +import javax.validation.Valid; import java.io.IOException; import java.util.List; @@ -44,6 +46,7 @@ import java.util.List; * @date 2022/2/23 18:26 **/ @Tag(name = "文件控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 4) @RestController @Validated public class DevFileController { @@ -63,6 +66,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 1) @Operation(summary = "动态上传文件返回id") @CommonLog("动态上传文件返回id") @PostMapping("/dev/file/uploadDynamicReturnId") @@ -76,6 +80,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 2) @Operation(summary = "动态上传文件返回url") @CommonLog("动态上传文件返回url") @PostMapping("/dev/file/uploadDynamicReturnUrl") @@ -89,6 +94,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 3) @Operation(summary = "上传本地文件返回id") @CommonLog("上传本地文件返回id") @PostMapping("/dev/file/uploadLocalReturnId") @@ -102,6 +108,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 4) @Operation(summary = "上传本地文件返回url") @CommonLog("上传本地文件返回url") @PostMapping("/dev/file/uploadLocalReturnUrl") @@ -115,6 +122,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 5) @Operation(summary = "上传阿里云文件返回id") @CommonLog("上传阿里云文件返回id") @PostMapping("/dev/file/uploadAliyunReturnId") @@ -128,6 +136,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 6) @Operation(summary = "上传阿里云文件返回url") @CommonLog("上传阿里云文件返回url") @PostMapping("/dev/file/uploadAliyunReturnUrl") @@ -141,6 +150,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 7) @Operation(summary = "上传腾讯云文件返回id") @CommonLog("上传腾讯云文件返回id") @PostMapping("/dev/file/uploadTencentReturnId") @@ -154,6 +164,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 8) @Operation(summary = "上传腾讯云文件返回url") @CommonLog("上传腾讯云文件返回url") @PostMapping("/dev/file/uploadTencentReturnUrl") @@ -167,6 +178,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 9) @Operation(summary = "上传MINIO文件返回id") @CommonLog("上传MINIO文件返回id") @PostMapping("/dev/file/uploadMinioReturnId") @@ -180,6 +192,7 @@ public class DevFileController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 10) @Operation(summary = "上传MINIO文件返回url") @CommonLog("上传MINIO文件返回url") @PostMapping("/dev/file/uploadMinioReturnUrl") @@ -193,6 +206,7 @@ public class DevFileController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 11) @Operation(summary = "获取文件分页列表") @GetMapping("/dev/file/page") public CommonResult> page(DevFilePageParam devFilePageParam) { @@ -205,6 +219,7 @@ public class DevFileController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 12) @Operation(summary = "获取文件列表") @GetMapping("/dev/file/list") public CommonResult> list(DevFileListParam devFileListParam) { @@ -217,6 +232,7 @@ public class DevFileController { * @author xuyuxiang * @date 2022/6/21 15:44 **/ + @ApiOperationSupport(order = 13) @Operation(summary = "下载文件") @CommonLog("下载文件") @GetMapping(value = "/dev/file/download", produces = MediaType.APPLICATION_OCTET_STREAM_VALUE) @@ -230,6 +246,7 @@ public class DevFileController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 14) @Operation(summary = "删除文件") @CommonLog("删除文件") @PostMapping(value = "/dev/file/delete") @@ -239,12 +256,27 @@ public class DevFileController { return CommonResult.ok(); } + /** + * 物理删除文件 + * + * @author 每天一点 + * @date 2025/4/06 20:25 + */ + @Operation(summary = "物理删除文件") + @CommonLog("物理删除文件") + @PostMapping(value = "/dev/file/deleteAbsolute") + public CommonResult deleteAbsolute(@RequestBody DevFileIdParam devFileIdParam) { + devFileService.deleteAbsolute(devFileIdParam); + return CommonResult.ok(); + } + /** * 获取文件详情 * * @author xuyuxiang * @date 2022/6/21 15:44 **/ + @ApiOperationSupport(order = 15) @Operation(summary = "获取文件详情") @GetMapping("/dev/file/detail") public CommonResult detail(@Valid DevFileIdParam devFileIdParam) { @@ -257,6 +289,7 @@ public class DevFileController { * @author yubaoshan * @date 2024/6/9 23:52 **/ + @ApiOperationSupport(order = 16) @Operation(summary = "根据文件url集合获取文件集合") @PostMapping("/dev/file/getFileListByUrlList") public CommonResult> getFileListByUrlList(@RequestBody @Valid DevFileUrlListParam devFileUrlListParam) { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/entity/DevFile.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/entity/DevFile.java index 1a456d33..0ddc6316 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/entity/DevFile.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/entity/DevFile.java @@ -30,7 +30,7 @@ import vip.xiaonuo.common.pojo.CommonEntity; public class DevFile extends CommonEntity { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 存储引擎 */ diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/param/DevFileIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/param/DevFileIdParam.java index e2e04c05..b546af26 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/param/DevFileIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/param/DevFileIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevFileIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/provider/DevFileApiProvider.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/provider/DevFileApiProvider.java index 4b25f3fe..e4927a68 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/provider/DevFileApiProvider.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/provider/DevFileApiProvider.java @@ -38,16 +38,21 @@ public class DevFileApiProvider implements DevFileApi { private static final String SNOWY_SYS_DEFAULT_FILE_ENGINE_KEY = "SNOWY_SYS_DEFAULT_FILE_ENGINE"; @Resource - private DevFileService devFileService; + private DevConfigApi devConfigApi; @Resource - private DevConfigApi devConfigApi; + private DevFileService devFileService; @Override public String uploadDynamicReturnId(MultipartFile file) { return devFileService.uploadReturnId(devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_FILE_ENGINE_KEY), file); } + @Override + public String uploadDynamicReturnUrl(MultipartFile file) { + return devFileService.uploadReturnUrl(devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_FILE_ENGINE_KEY), file); + } + @Override public String storageFileWithReturnUrlLocal(MultipartFile file) { return devFileService.uploadReturnUrl(DevFileEngineTypeEnum.LOCAL.getValue(), file); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/service/impl/DevFileServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/service/impl/DevFileServiceImpl.java index c6360197..5f8ec4c9 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/service/impl/DevFileServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/service/impl/DevFileServiceImpl.java @@ -124,8 +124,8 @@ public class DevFileServiceImpl extends ServiceImpl impl CommonDownloadUtil.download(devFile.getName(), IoUtil.readBytes(FileUtil.getInputStream(file)), response); } - @Override @Transactional(rollbackFor = Exception.class) + @Override public void delete(List devFileIdParamList) { this.removeByIds(CollStreamUtil.toList(devFileIdParamList, DevFileIdParam::getId)); } @@ -178,6 +178,8 @@ public class DevFileServiceImpl extends ServiceImpl impl // 存储桶名称 String bucketName; + String fileKey = genFileKey(fileId, file); + // 定义存储的url,本地文件返回文件实际路径,其他引擎返回网络地址 String storageUrl; @@ -186,22 +188,22 @@ public class DevFileServiceImpl extends ServiceImpl impl // 使用固定名称defaultBucketName bucketName = "defaultBucketName"; - storageUrl = DevFileLocalUtil.storageFileWithReturnUrl(bucketName, genFileKey(fileId, file), file); + storageUrl = DevFileLocalUtil.storageFileWithReturnUrl(bucketName, fileKey, file); } else if(engine.equals(DevFileEngineTypeEnum.ALIYUN.getValue())) { // 使用阿里云默认配置的bucketName bucketName = DevFileAliyunUtil.getDefaultBucketName(); - storageUrl = DevFileAliyunUtil.storageFileWithReturnUrl(bucketName, genFileKey(fileId, file), file); + storageUrl = DevFileAliyunUtil.storageFileWithReturnUrl(bucketName, fileKey, file); } else if(engine.equals(DevFileEngineTypeEnum.TENCENT.getValue())) { // 使用腾讯云默认配置的bucketName bucketName = DevFileTencentUtil.getDefaultBucketName(); - storageUrl = DevFileTencentUtil.storageFileWithReturnUrl(bucketName, genFileKey(fileId, file), file); + storageUrl = DevFileTencentUtil.storageFileWithReturnUrl(bucketName, fileKey, file); } else if(engine.equals(DevFileEngineTypeEnum.MINIO.getValue())) { // 使用MINIO默认配置的bucketName bucketName = DevFileMinIoUtil.getDefaultBucketName(); - storageUrl = DevFileMinIoUtil.storageFileWithReturnUrl(bucketName, genFileKey(fileId, file), file); + storageUrl = DevFileMinIoUtil.storageFileWithReturnUrl(bucketName, fileKey, file); } else { throw new CommonException("不支持的文件引擎:{}", engine); } @@ -215,12 +217,13 @@ public class DevFileServiceImpl extends ServiceImpl impl // 设置存储引擎类型 devFile.setEngine(engine); devFile.setBucket(bucketName); + devFile.setFileKey(fileKey); devFile.setName(file.getOriginalFilename()); String suffix = ObjectUtil.isNotEmpty(file.getOriginalFilename())?StrUtil.subAfter(file.getOriginalFilename(), StrUtil.DOT, true):null; devFile.setSuffix(suffix); devFile.setSizeKb(Convert.toStr(NumberUtil.div(new BigDecimal(file.getSize()), BigDecimal.valueOf(1024)) - .setScale(0, RoundingMode.HALF_UP))); + .setScale(0, RoundingMode.HALF_UP ))); devFile.setSizeInfo(FileUtil.readableFileSize(file.getSize())); devFile.setObjName(ObjectUtil.isNotEmpty(devFile.getSuffix())?fileId + StrUtil.DOT + devFile.getSuffix():null); // 如果是图片,则压缩生成缩略图 diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileAliyunUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileAliyunUtil.java index 7a304ca0..174490e2 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileAliyunUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileAliyunUtil.java @@ -23,13 +23,13 @@ import com.aliyun.oss.model.CannedAccessControlList; import com.aliyun.oss.model.GeneratePresignedUrlRequest; import com.aliyun.oss.model.OSSObject; import com.aliyun.oss.model.ObjectMetadata; -import jakarta.activation.MimetypesFileTypeMap; import lombok.extern.slf4j.Slf4j; import org.springframework.web.multipart.MultipartFile; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.dev.api.DevConfigApi; import vip.xiaonuo.dev.modular.file.enums.DevFileBucketAuthEnum; +import javax.activation.MimetypesFileTypeMap; import java.io.*; import java.net.URL; import java.util.Date; @@ -37,7 +37,7 @@ import java.util.List; /** * 阿里云文件工具类 - * 参考文档:https://help.aliyun.com/document_detail/32010.html + * 参考文档:参考文档 * * @author xuyuxiang * @date 2022/1/2 18:13 diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileLocalUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileLocalUtil.java index 6bba1924..7dec4e72 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileLocalUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileLocalUtil.java @@ -19,6 +19,7 @@ import cn.hutool.extra.spring.SpringUtil; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; import cn.hutool.system.SystemUtil; +import lombok.Getter; import lombok.extern.slf4j.Slf4j; import org.springframework.web.multipart.MultipartFile; import vip.xiaonuo.common.exception.CommonException; @@ -39,6 +40,12 @@ import java.io.InputStream; @Slf4j public class DevFileLocalUtil { + /** + * -- GETTER -- + * 获取操作的客户端 + * + */ + @Getter private static JSONObject client; private static final String SNOWY_FILE_LOCAL_FOLDER_FOR_WINDOWS_KEY = "SNOWY_FILE_LOCAL_FOLDER_FOR_WINDOWS"; @@ -92,16 +99,6 @@ public class DevFileLocalUtil { client.clear(); } - /** - * 获取操作的客户端 - * - * @author xuyuxiang - * @date 2022/1/5 23:24 - */ - public static JSONObject getClient() { - return client; - } - /** * 获取上传地址 * diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileMinIoUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileMinIoUtil.java index 0b4054db..91af4d54 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileMinIoUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileMinIoUtil.java @@ -23,18 +23,19 @@ import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; import io.minio.*; import io.minio.http.Method; -import jakarta.activation.MimetypesFileTypeMap; +import lombok.Getter; import lombok.extern.slf4j.Slf4j; import org.springframework.web.multipart.MultipartFile; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.dev.api.DevConfigApi; import vip.xiaonuo.dev.modular.file.enums.DevFileBucketAuthEnum; +import javax.activation.MimetypesFileTypeMap; import java.io.*; /** * MINIO文件工具类 - * 参考文档:http://docs.minio.org.cn/docs/master/java-client-quickstart-guide + * 参考文档:参考文档 * * @author xuyuxiang * @date 2022/1/2 18:13 @@ -42,6 +43,12 @@ import java.io.*; @Slf4j public class DevFileMinIoUtil { + /** + * -- GETTER -- + * 获取操作的客户端 + * + */ + @Getter private static MinioClient client; private static String defaultBucketName; @@ -80,7 +87,7 @@ public class DevFileMinIoUtil { String endpoint = devConfigApi.getValueByKey(SNOWY_FILE_MINIO_END_POINT_KEY); if(ObjectUtil.isEmpty(endpoint)) { - throw new CommonException("MINIO文件操作客户端未正确配置:endpoint为空"); + throw new CommonException("MINIO文件操作客户端未正确配置:secretKey为空"); } /* 默认BucketName */ @@ -114,16 +121,6 @@ public class DevFileMinIoUtil { // 无需 } - /** - * 获取操作的客户端 - * - * @author xuyuxiang - * @date 2022/1/5 23:24 - */ - public static MinioClient getClient() { - return client; - } - /** * 查询存储桶是否存在 * 例如:传入参数examplebucket-1250000000,返回true代表存在此桶 diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileTencentUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileTencentUtil.java index 33f82f6c..cdd050b5 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileTencentUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/file/util/DevFileTencentUtil.java @@ -27,13 +27,13 @@ import com.qcloud.cos.model.*; import com.qcloud.cos.region.Region; import com.qcloud.cos.transfer.TransferManager; import com.qcloud.cos.transfer.TransferManagerConfiguration; -import jakarta.activation.MimetypesFileTypeMap; import lombok.extern.slf4j.Slf4j; import org.springframework.web.multipart.MultipartFile; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.dev.api.DevConfigApi; import vip.xiaonuo.dev.modular.file.enums.DevFileBucketAuthEnum; +import javax.activation.MimetypesFileTypeMap; import java.io.*; import java.net.URL; import java.util.Date; @@ -42,7 +42,7 @@ import java.util.concurrent.Executors; /** * 腾讯云文件工具类 - * 参考文档:https://cloud.tencent.com/document/product/436/10199 + * 参考文档:参考文档 * * @author xuyuxiang * @date 2022/1/2 18:13 diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/controller/DevJobController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/controller/DevJobController.java index ebd0e9b4..619e267b 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/controller/DevJobController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/controller/DevJobController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.dev.modular.job.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -29,6 +30,7 @@ import vip.xiaonuo.dev.modular.job.entity.DevJob; import vip.xiaonuo.dev.modular.job.param.*; import vip.xiaonuo.dev.modular.job.service.DevJobService; +import javax.validation.Valid; import java.util.List; /** @@ -38,19 +40,21 @@ import java.util.List; * @date 2022/8/5 10:48 **/ @Tag(name = "定时任务控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 7) @RestController @Validated public class DevJobController { @Resource private DevJobService devJobService; - + /** * 获取定时任务分页 * * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取定时任务分页") @GetMapping("/dev/job/page") public CommonResult> page(DevJobPageParam devJobPageParam) { @@ -63,6 +67,7 @@ public class DevJobController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取定时任务列表") @GetMapping("/dev/job/list") public CommonResult> list(DevJobListParam devJobListParam) { @@ -75,6 +80,7 @@ public class DevJobController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "添加定时任务") @CommonLog("添加定时任务") @PostMapping("/dev/job/add") @@ -89,6 +95,7 @@ public class DevJobController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "编辑定时任务") @CommonLog("编辑定时任务") @PostMapping("/dev/job/edit") @@ -103,6 +110,7 @@ public class DevJobController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "删除定时任务") @CommonLog("删除定时任务") @PostMapping("/dev/job/delete") @@ -118,6 +126,7 @@ public class DevJobController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取定时任务详情") @GetMapping("/dev/job/detail") public CommonResult detail(@Valid DevJobIdParam devJobIdParam) { @@ -130,6 +139,7 @@ public class DevJobController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 6) @Operation(summary = "停止定时任务") @CommonLog("停止定时任务") @PostMapping("/dev/job/stopJob") @@ -144,6 +154,7 @@ public class DevJobController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 7) @Operation(summary = "运行定时任务") @CommonLog("运行定时任务") @PostMapping("/dev/job/runJob") @@ -158,6 +169,7 @@ public class DevJobController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 8) @Operation(summary = "立即运行定时任务") @CommonLog("立即运行定时任务") @PostMapping("/dev/job/runJobNow") @@ -172,6 +184,7 @@ public class DevJobController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 9) @Operation(summary = "获取定时任务类") @GetMapping("/dev/job/getActionClass") public CommonResult> getActionClass() { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/entity/DevJob.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/entity/DevJob.java index a927c92f..6328fde4 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/entity/DevJob.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/entity/DevJob.java @@ -65,7 +65,7 @@ public class DevJob extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobAddParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobAddParam.java index 509dc076..69e35833 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobAddParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobAddParam.java @@ -29,27 +29,27 @@ import lombok.Setter; public class DevJobAddParam { /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 任务类名 */ - @Schema(description = "任务类名", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "任务类名") @NotBlank(message = "actionClass不能为空") private String actionClass; /** cron表达式 */ - @Schema(description = "cron表达式", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "cron表达式") @NotBlank(message = "cronExpression不能为空") private String cronExpression; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobEditParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobEditParam.java index 5a8824df..25e61840 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobEditParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobEditParam.java @@ -29,32 +29,32 @@ import lombok.Setter; public class DevJobEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 任务类名 */ - @Schema(description = "任务类名", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "任务类名") @NotBlank(message = "actionClass不能为空") private String actionClass; /** cron表达式 */ - @Schema(description = "cron表达式", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "cron表达式") @NotBlank(message = "cronExpression不能为空") private String cronExpression; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobIdParam.java index f2e7f987..991cdbbd 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/param/DevJobIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevJobIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/service/impl/DevJobServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/service/impl/DevJobServiceImpl.java index 264e3a2f..de84657d 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/service/impl/DevJobServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/service/impl/DevJobServiceImpl.java @@ -82,20 +82,20 @@ public class DevJobServiceImpl extends ServiceImpl impleme @Override public List list(DevJobListParam devJobListParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 - lambdaQueryWrapper.select(DevJob::getId, DevJob::getName, DevJob::getCategory, + queryWrapper.lambda().select(DevJob::getId, DevJob::getName, DevJob::getCategory, DevJob::getActionClass, DevJob::getCronExpression, DevJob::getJobStatus, DevJob::getSortCode); if(ObjectUtil.isNotEmpty(devJobListParam.getCategory())) { - lambdaQueryWrapper.eq(DevJob::getCategory, devJobListParam.getCategory()); + queryWrapper.lambda().eq(DevJob::getCategory, devJobListParam.getCategory()); } if(ObjectUtil.isNotEmpty(devJobListParam.getSearchKey())) { - lambdaQueryWrapper.like(DevJob::getName, devJobListParam.getSearchKey()); + queryWrapper.lambda().like(DevJob::getName, devJobListParam.getSearchKey()); } if(ObjectUtil.isNotEmpty(devJobListParam.getJobStatus())) { - lambdaQueryWrapper.like(DevJob::getJobStatus, devJobListParam.getJobStatus()); + queryWrapper.lambda().like(DevJob::getJobStatus, devJobListParam.getJobStatus()); } - return this.list(lambdaQueryWrapper); + return this.list(queryWrapper); } @Override @@ -223,6 +223,7 @@ public class DevJobServiceImpl extends ServiceImpl impleme .set(DevJob::getJobStatus, DevJobStatusEnum.RUNNING.getValue())); } + @Transactional(rollbackFor = Exception.class) @Override public void runJobNow(DevJobIdParam devJobIdParam) { DevJob devJob = this.detail(devJobIdParam); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/task/DevJobTimerTaskRunner.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/task/DevJobTimerTaskRunner.java index 98ddefd1..c97b35b4 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/task/DevJobTimerTaskRunner.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/job/task/DevJobTimerTaskRunner.java @@ -30,7 +30,7 @@ public class DevJobTimerTaskRunner implements CommonTimerTaskRunner { @Override public void action(String extJson) { - log.info("我是一个定时任务,正在在被执行第" + n + "次"); + log.info("我是一个定时任务,正在在被执行第{}次", n); n = n + 1; } } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/controller/DevLogController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/controller/DevLogController.java index ee7d7dec..28bdd870 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/controller/DevLogController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/controller/DevLogController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.dev.modular.log.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; @@ -34,6 +35,7 @@ import vip.xiaonuo.dev.modular.log.result.DevLogVisLineChartDataResult; import vip.xiaonuo.dev.modular.log.result.DevLogVisPieChartDataResult; import vip.xiaonuo.dev.modular.log.service.DevLogService; +import javax.validation.Valid; import java.util.List; /** @@ -43,6 +45,7 @@ import java.util.List; * @date 2022/9/2 15:15 */ @Tag(name = "日志控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 8) @RestController @Validated public class DevLogController { @@ -56,6 +59,7 @@ public class DevLogController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取日志分页") @GetMapping("/dev/log/page") public CommonResult> page(DevLogPageParam devLogPageParam) { @@ -68,6 +72,7 @@ public class DevLogController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "清空日志") @CommonLog("清空日志") @PostMapping("/dev/log/delete") @@ -82,6 +87,7 @@ public class DevLogController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 3) @Operation(summary = "获取访问日志折线图数据") @GetMapping("/dev/log/vis/lineChartData") public CommonResult> visLogLineChartData() { @@ -94,6 +100,7 @@ public class DevLogController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "获取访问日志饼状图数据") @GetMapping("/dev/log/vis/pieChartData") public CommonResult> visLogPieChartData() { @@ -106,6 +113,7 @@ public class DevLogController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取操作日志柱状图数据") @GetMapping("/dev/log/op/barChartData") public CommonResult> opLogBarChartData() { @@ -118,6 +126,7 @@ public class DevLogController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取操作日志饼状图数据") @GetMapping("/dev/log/op/pieChartData") public CommonResult> opLogPieChartData() { @@ -127,6 +136,7 @@ public class DevLogController { /** * 依据id获取单条日志详情 */ + @ApiOperationSupport(order = 6) @Operation(summary = "依据id获取日志详情") @GetMapping("/dev/log/detail") public CommonResult detail(@Valid DevLogIdParam devLogIdParam) { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/entity/DevLog.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/entity/DevLog.java index 22628439..b2d95bde 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/entity/DevLog.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/entity/DevLog.java @@ -33,7 +33,7 @@ import java.util.Date; public class DevLog { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 日志分类 */ diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogDeleteParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogDeleteParam.java index b688e613..438c287f 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogDeleteParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogDeleteParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevLogDeleteParam { /** 日志分类 */ - @Schema(description = "日志分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "日志分类") @NotBlank(message = "category不能为空") private String category; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogIdParam.java index 33858a5c..c00639f2 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/param/DevLogIdParam.java @@ -12,7 +12,7 @@ import lombok.Setter; @Setter public class DevLogIdParam { - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/service/impl/DevLogServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/service/impl/DevLogServiceImpl.java index 41b5c5cf..15f2d431 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/service/impl/DevLogServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/service/impl/DevLogServiceImpl.java @@ -148,7 +148,7 @@ public class DevLogServiceImpl extends ServiceImpl impleme DevLog::getName,DevLog::getOpIp, DevLog::getOpAddress, DevLog::getCategory, DevLog::getClassName, DevLog::getMethodName, DevLog::getOpTime, DevLog::getOpUser) .in(DevLog::getCategory, DevLogCategoryEnum.OPERATE.getValue(), - DevLogCategoryEnum.EXCEPTION.getValue()).between(DevLog::getOpTime, lastWeek, now).orderByAsc(DevLog::getOpTime)) + DevLogCategoryEnum.EXCEPTION.getValue()).between(DevLog::getOpTime, lastWeek, now).orderByAsc(DevLog::getOpTime)) .stream().map(devLog -> JSONUtil.parseObj(devLog).set("date", DateUtil.formatDate(devLog.getOpTime()))) .collect(Collectors.groupingBy(jsonObject -> jsonObject.getStr("date"))); long between = DateUtil.between(lastWeek, now, DateUnit.DAY); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/util/DevLogUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/util/DevLogUtil.java index 5525b495..f161d289 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/util/DevLogUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/log/util/DevLogUtil.java @@ -12,11 +12,9 @@ */ package vip.xiaonuo.dev.modular.log.util; -import cn.dev33.satoken.stp.StpUtil; import cn.hutool.core.date.DateTime; import cn.hutool.core.exceptions.ExceptionUtil; import cn.hutool.core.thread.ThreadUtil; -import cn.hutool.core.util.ObjectUtil; import cn.hutool.extra.spring.SpringUtil; import jakarta.servlet.http.HttpServletRequest; import org.aspectj.lang.JoinPoint; @@ -140,21 +138,11 @@ public class DevLogUtil { private static DevLog genBasOpLog() { HttpServletRequest request = CommonServletUtil.getRequest(); String ip = CommonIpAddressUtil.getIp(request); - String loginId; - try { - loginId = StpUtil.getLoginIdAsString(); - if (ObjectUtil.isEmpty(loginId)) { - loginId = "-1"; - } - } catch (Exception e) { - loginId = "-1"; - } DevLog devLog = new DevLog(); devLog.setOpIp(CommonIpAddressUtil.getIp(request)); devLog.setOpAddress(CommonIpAddressUtil.getCityInfo(ip)); devLog.setOpBrowser(CommonUaUtil.getBrowser(request)); devLog.setOpOs(CommonUaUtil.getOs(request)); - devLog.setCreateUser(loginId); return devLog; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/controller/DevMessageController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/controller/DevMessageController.java index 66a6ae0d..f6462c0e 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/controller/DevMessageController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/controller/DevMessageController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.dev.modular.message.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -32,6 +33,7 @@ import vip.xiaonuo.dev.modular.message.param.DevMessageSendParam; import vip.xiaonuo.dev.modular.message.result.DevMessageResult; import vip.xiaonuo.dev.modular.message.service.DevMessageService; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +43,7 @@ import java.util.List; * @date 2022/6/21 14:57 **/ @Tag(name = "站内信控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 6) @RestController @Validated public class DevMessageController { @@ -54,6 +57,7 @@ public class DevMessageController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 1) @Operation(summary = "发送站内信") @CommonLog("发送站内信") @PostMapping("/dev/message/send") @@ -68,6 +72,7 @@ public class DevMessageController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取站内信分页") @GetMapping("/dev/message/page") public CommonResult> page(DevMessagePageParam devMessagePageParam) { @@ -80,11 +85,12 @@ public class DevMessageController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 3) @Operation(summary = "删除站内信") @CommonLog("删除站内信") @PostMapping("/dev/message/delete") public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List devMessageIdParamList) { + List devMessageIdParamList) { devMessageService.delete(devMessageIdParamList); return CommonResult.ok(); } @@ -95,6 +101,7 @@ public class DevMessageController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "获取站内信详情") @GetMapping("/dev/message/detail") public CommonResult detail(@Valid DevMessageIdParam devMessageIdParam) { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/entity/DevMessage.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/entity/DevMessage.java index 8d3a88b4..3275adcc 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/entity/DevMessage.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/entity/DevMessage.java @@ -30,7 +30,7 @@ import vip.xiaonuo.common.pojo.CommonEntity; public class DevMessage extends CommonEntity { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 分类 */ diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageIdParam.java index cd7164e4..aa98ef10 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevMessageIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageSendParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageSendParam.java index d304144c..a5d8e72a 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageSendParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/param/DevMessageSendParam.java @@ -31,12 +31,12 @@ import java.util.List; public class DevMessageSendParam { /** 主题 */ - @Schema(description = "主题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "主题") @NotBlank(message = "subject不能为空") private String subject; /** 接收人id集合 */ - @Schema(description = "接收人id集合", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "接收人id集合") @NotEmpty(message = "receiverIdList不能为空") private List receiverIdList; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/result/DevMessageResult.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/result/DevMessageResult.java index 0f0f636c..48e2076c 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/result/DevMessageResult.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/result/DevMessageResult.java @@ -29,7 +29,7 @@ import java.util.List; public class DevMessageResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 分类 */ diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/service/impl/DevMessageServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/service/impl/DevMessageServiceImpl.java index a55c59c1..04407bf4 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/service/impl/DevMessageServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/message/service/impl/DevMessageServiceImpl.java @@ -192,13 +192,7 @@ public class DevMessageServiceImpl extends ServiceImpl receiveInfoList = devRelationService.getRelationListByObjectIdAndCategory(devMessage.getId(), DevRelationCategoryEnum.MSG_TO_USER.getValue()).stream().map(devRelation -> { DevMessageResult.DevReceiveInfo devReceiveInfo = new DevMessageResult.DevReceiveInfo(); - JSONObject userObj = null; - try { - userObj = sysUserApi.getUserByIdWithException(devRelation.getTargetId()); - } - catch (Exception e) { - // 收件人中包含删除用户 在此处做处理 - } + JSONObject userObj = sysUserApi.getUserByIdWithException(devRelation.getTargetId()); String userName = "未知用户"; if(ObjectUtil.isNotEmpty(userObj)) { userName = userObj.getStr("name"); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/monitor/controller/DevMonitorController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/monitor/controller/DevMonitorController.java index 1b6d6eb5..cf7fa4c8 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/monitor/controller/DevMonitorController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/monitor/controller/DevMonitorController.java @@ -12,6 +12,8 @@ */ package vip.xiaonuo.dev.modular.monitor.controller; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; @@ -29,6 +31,7 @@ import vip.xiaonuo.dev.modular.monitor.service.DevMonitorService; * @date 2022/6/21 14:57 **/ @Tag(name = "监控控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 9) @RestController @Validated public class DevMonitorController { @@ -42,6 +45,7 @@ public class DevMonitorController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取服务器监控信息") @GetMapping("/dev/monitor/serverInfo") public CommonResult serverInfo() { @@ -54,6 +58,7 @@ public class DevMonitorController { * @author diantu * @date 2023/7/27 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取服务器网络情况") @GetMapping("/dev/monitor/networkInfo") public CommonResult networkInfo() { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/controller/DevWeakPasswordController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/controller/DevWeakPasswordController.java new file mode 100644 index 00000000..41130fc6 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/controller/DevWeakPasswordController.java @@ -0,0 +1,125 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.controller; + +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.annotation.Resource; +import jakarta.validation.constraints.NotEmpty; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RestController; +import vip.xiaonuo.common.annotation.CommonLog; +import vip.xiaonuo.common.pojo.CommonResult; +import vip.xiaonuo.dev.modular.password.entity.DevWeakPassword; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordAddParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordEditParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordIdParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordPageParam; +import vip.xiaonuo.dev.modular.password.service.DevWeakPasswordService; + +import javax.validation.Valid; +import java.util.List; + +/** + * 弱密码库控制器 + * + * @author xuyuxiang + * @date 2022/4/25 20:40 + */ +@Tag(name = "弱密码库控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 2) +@RestController +@Validated +public class DevWeakPasswordController { + + @Resource + private DevWeakPasswordService devWeakPasswordService; + + /** + * 获取弱密码库分页 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 1) + @Operation(summary = "获取弱密码库分页") + @GetMapping("/dev/weakPassword/page") + public CommonResult> page(DevWeakPasswordPageParam devWeakPasswordPageParam) { + return CommonResult.data(devWeakPasswordService.page(devWeakPasswordPageParam)); + } + + /** + * 添加弱密码库 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 2) + @Operation(summary = "添加弱密码库") + @CommonLog("添加弱密码库") + @PostMapping("/dev/weakPassword/add") + public CommonResult add(@RequestBody @Valid DevWeakPasswordAddParam devWeakPasswordAddParam) { + devWeakPasswordService.add(devWeakPasswordAddParam); + return CommonResult.ok(); + } + + /** + * 编辑弱密码库 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 3) + @Operation(summary = "编辑弱密码库") + @CommonLog("编辑弱密码库") + @PostMapping("/dev/weakPassword/edit") + public CommonResult edit(@RequestBody @Valid DevWeakPasswordEditParam devWeakPasswordEditParam) { + devWeakPasswordService.edit(devWeakPasswordEditParam); + return CommonResult.ok(); + } + + /** + * 删除弱密码库 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 4) + @Operation(summary = "删除弱密码库") + @CommonLog("删除弱密码库") + @PostMapping("/dev/weakPassword/delete") + public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") + List devWeakPasswordIdParamList) { + devWeakPasswordService.delete(devWeakPasswordIdParamList); + return CommonResult.ok(); + } + + /** + * 获取弱密码库详情 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 5) + @Operation(summary = "获取弱密码库详情") + @GetMapping("/dev/weakPassword/detail") + public CommonResult detail(@Valid DevWeakPasswordIdParam devWeakPasswordIdParam) { + return CommonResult.data(devWeakPasswordService.detail(devWeakPasswordIdParam)); + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/entity/DevWeakPassword.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/entity/DevWeakPassword.java new file mode 100644 index 00000000..23ad936c --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/entity/DevWeakPassword.java @@ -0,0 +1,42 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +/** + * 弱密码库实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("DEV_WEAK_PASSWORD") +public class DevWeakPassword extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 密码 */ + @Schema(description = "密码") + private String password; + +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/mapper/DevWeakPasswordMapper.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/mapper/DevWeakPasswordMapper.java new file mode 100644 index 00000000..108da3b4 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/mapper/DevWeakPasswordMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.dev.modular.password.entity.DevWeakPassword; + +/** + * 弱密码库Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface DevWeakPasswordMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/mapper/mapping/DevWeakPasswordMapper.xml b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/mapper/mapping/DevWeakPasswordMapper.xml new file mode 100644 index 00000000..261af514 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/mapper/mapping/DevWeakPasswordMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordAddParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordAddParam.java new file mode 100644 index 00000000..64aa3e07 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordAddParam.java @@ -0,0 +1,34 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 弱密码库添加参数 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +public class DevWeakPasswordAddParam { + + /** 密码 */ + @Schema(description = "密码") + @NotBlank(message = "password不能为空") + private String password; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordEditParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordEditParam.java new file mode 100644 index 00000000..2a3c67ca --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordEditParam.java @@ -0,0 +1,39 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 弱密码库编辑参数 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +public class DevWeakPasswordEditParam { + + /** id */ + @Schema(description = "id") + @NotBlank(message = "id不能为空") + private String id; + + /** 密码 */ + @Schema(description = "密码") + @NotBlank(message = "password不能为空") + private String password; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordIdParam.java new file mode 100644 index 00000000..7602c481 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordIdParam.java @@ -0,0 +1,34 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 弱密码库Id参数 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +public class DevWeakPasswordIdParam { + + /** id */ + @Schema(description = "id") + @NotBlank(message = "id不能为空") + private String id; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordPageParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordPageParam.java new file mode 100644 index 00000000..6a00fa43 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/param/DevWeakPasswordPageParam.java @@ -0,0 +1,48 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; + +/** + * 弱密码库查询参数 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +public class DevWeakPasswordPageParam { + + /** 当前页 */ + @Schema(description = "当前页码") + private Integer current; + + /** 每页条数 */ + @Schema(description = "每页条数") + private Integer size; + + /** 排序字段 */ + @Schema(description = "排序字段,字段驼峰名称,如:userName") + private String sortField; + + /** 排序方式 */ + @Schema(description = "排序方式,升序:ASCEND;降序:DESCEND") + private String sortOrder; + + /** 密码关键词 */ + @Schema(description = "密码关键词") + private String searchKey; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/provider/DevWeakPasswordApiProvider.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/provider/DevWeakPasswordApiProvider.java new file mode 100644 index 00000000..1e50ef02 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/provider/DevWeakPasswordApiProvider.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.provider; + +import jakarta.annotation.Resource; +import org.springframework.stereotype.Service; +import vip.xiaonuo.dev.api.DevWeakPasswordApi; +import vip.xiaonuo.dev.modular.password.entity.DevWeakPassword; +import vip.xiaonuo.dev.modular.password.service.DevWeakPasswordService; + +import java.util.List; +import java.util.stream.Collectors; + +/** + * 弱密码库API接口实现类 + * + * @author xuyuxiang + * @date 2022/6/17 14:43 + **/ +@Service +public class DevWeakPasswordApiProvider implements DevWeakPasswordApi { + + @Resource + private DevWeakPasswordService devWeakPasswordService; + + @Override + public List weakPasswordList() { + return devWeakPasswordService.list().stream().map(DevWeakPassword::getPassword).collect(Collectors.toList()); + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/service/DevWeakPasswordService.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/service/DevWeakPasswordService.java new file mode 100644 index 00000000..cfad0709 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/service/DevWeakPasswordService.java @@ -0,0 +1,81 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.service; + +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.dev.modular.password.entity.DevWeakPassword; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordAddParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordEditParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordIdParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordPageParam; + +import java.util.List; + +/** + * 弱密码库Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface DevWeakPasswordService extends IService { + + /** + * 获取弱密码库分页 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + Page page(DevWeakPasswordPageParam devWeakPasswordPageParam); + + /** + * 添加弱密码库 + * + * @author xuyuxiang + * @date 2022/4/24 20:48 + */ + void add(DevWeakPasswordAddParam devWeakPasswordAddParam); + + /** + * 编辑弱密码库 + * + * @author xuyuxiang + * @date 2022/4/24 21:13 + */ + void edit(DevWeakPasswordEditParam devWeakPasswordEditParam); + + /** + * 删除弱密码库 + * + * @author xuyuxiang + * @date 2022/4/24 21:18 + */ + void delete(List devWeakPasswordIdParamList); + + /** + * 获取弱密码库详情 + * + * @author xuyuxiang + * @date 2022/4/24 21:18 + */ + DevWeakPassword detail(DevWeakPasswordIdParam devWeakPasswordIdParam); + + /** + * 获取弱密码库详情 + * + * @author xuyuxiang + * @date 2022/7/25 19:42 + **/ + DevWeakPassword queryEntity(String id); + +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/service/impl/DevWeakPasswordServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/service/impl/DevWeakPasswordServiceImpl.java new file mode 100644 index 00000000..fef62084 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/password/service/impl/DevWeakPasswordServiceImpl.java @@ -0,0 +1,108 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.password.service.impl; + +import cn.hutool.core.bean.BeanUtil; +import cn.hutool.core.collection.CollStreamUtil; +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import vip.xiaonuo.common.enums.CommonSortOrderEnum; +import vip.xiaonuo.common.exception.CommonException; +import vip.xiaonuo.common.page.CommonPageRequest; +import vip.xiaonuo.dev.modular.password.entity.DevWeakPassword; +import vip.xiaonuo.dev.modular.password.mapper.DevWeakPasswordMapper; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordAddParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordEditParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordIdParam; +import vip.xiaonuo.dev.modular.password.param.DevWeakPasswordPageParam; +import vip.xiaonuo.dev.modular.password.service.DevWeakPasswordService; + +import java.util.List; + +/** + * 弱密码库Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class DevWeakPasswordServiceImpl extends ServiceImpl implements DevWeakPasswordService { + + @Override + public Page page(DevWeakPasswordPageParam devWeakPasswordPageParam) { + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + if(ObjectUtil.isNotEmpty(devWeakPasswordPageParam.getSearchKey())) { + queryWrapper.lambda().like(DevWeakPassword::getPassword, devWeakPasswordPageParam.getSearchKey()); + } + if(ObjectUtil.isAllNotEmpty(devWeakPasswordPageParam.getSortField(), devWeakPasswordPageParam.getSortOrder())) { + CommonSortOrderEnum.validate(devWeakPasswordPageParam.getSortOrder()); + queryWrapper.orderBy(true, devWeakPasswordPageParam.getSortOrder().equals(CommonSortOrderEnum.ASC.getValue()), + StrUtil.toUnderlineCase(devWeakPasswordPageParam.getSortField())); + } else { + queryWrapper.lambda().orderByDesc(DevWeakPassword::getCreateTime); + } + return this.page(CommonPageRequest.defaultPage(), queryWrapper); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void add(DevWeakPasswordAddParam devWeakPasswordAddParam) { + DevWeakPassword devWeakPassword = BeanUtil.toBean(devWeakPasswordAddParam, DevWeakPassword.class); + boolean repeatPassword = this.count(new LambdaQueryWrapper().eq(DevWeakPassword::getPassword, devWeakPassword.getPassword())) > 0; + if(repeatPassword) { + throw new CommonException("存在重复的密码,值为:{}", devWeakPassword.getPassword()); + } + this.save(devWeakPassword); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void edit(DevWeakPasswordEditParam devWeakPasswordEditParam) { + DevWeakPassword devWeakPassword = this.queryEntity(devWeakPasswordEditParam.getId()); + BeanUtil.copyProperties(devWeakPasswordEditParam, devWeakPassword); + boolean repeatPassword = this.count(new LambdaQueryWrapper().eq(DevWeakPassword::getPassword, devWeakPassword.getPassword()) + .ne(DevWeakPassword::getId, devWeakPassword.getId())) > 0; + if(repeatPassword) { + throw new CommonException("存在重复的密码,值为:{}", devWeakPassword.getPassword()); + } + this.updateById(devWeakPassword); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void delete(List devWeakPasswordIdParamList) { + List weakPasswordIdList = CollStreamUtil.toList(devWeakPasswordIdParamList, DevWeakPasswordIdParam::getId); + // 执行删除 + this.removeByIds(weakPasswordIdList); + } + + @Override + public DevWeakPassword detail(DevWeakPasswordIdParam devWeakPasswordIdParam) { + return this.queryEntity(devWeakPasswordIdParam.getId()); + } + + @Override + public DevWeakPassword queryEntity(String id) { + DevWeakPassword devWeakPassword = this.getById(id); + if (ObjectUtil.isEmpty(devWeakPassword)) { + throw new CommonException("弱密码数据不存在,id值为:{}", id); + } + return devWeakPassword; + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/controller/DevPushController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/controller/DevPushController.java new file mode 100644 index 00000000..83c70a77 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/controller/DevPushController.java @@ -0,0 +1,212 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.controller; + +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.annotation.Resource; +import jakarta.validation.constraints.NotEmpty; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RestController; +import vip.xiaonuo.common.annotation.CommonLog; +import vip.xiaonuo.common.pojo.CommonResult; +import vip.xiaonuo.dev.modular.push.entity.DevPush; +import vip.xiaonuo.dev.modular.push.param.*; +import vip.xiaonuo.dev.modular.push.service.DevPushService; + +import javax.validation.Valid; +import java.util.List; + +/** + * 消息推送控制器 + * + * @author xuyuxiang + * @date 2022/2/23 18:26 + **/ +@Tag(name = "消息推送控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 11) +@RestController +@Validated +public class DevPushController { + + @Resource + private DevPushService devPushService; + + /** + * 动态推送消息 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 1) + @Operation(summary = "动态推送TEXT消息") + @CommonLog("动态推送TEXT消息") + @PostMapping("/dev/push/pushDynamicText") + public CommonResult pushDynamicText(@RequestBody @Valid DevPushDynamicTextParam devPushDynamicTextParam) { + devPushService.pushDynamicText(devPushDynamicTextParam); + return CommonResult.ok(); + } + + /** + * 推送消息——飞书TXT + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 2) + @Operation(summary = "推送飞书TEXT消息") + @CommonLog("推送飞书TEXT消息") + @PostMapping("/dev/push/pushFeiShuText") + public CommonResult pushFeiShuText(@RequestBody @Valid DevPushFeiShuTextParam devPushFeiShuTextParam) { + devPushService.pushFeiShuText(devPushFeiShuTextParam); + return CommonResult.ok(); + } + + /** + * 推送消息——钉钉TXT + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 3) + @Operation(summary = "推送钉钉TEXT消息") + @CommonLog("推送钉钉TEXT消息") + @PostMapping("/dev/push/pushDingTalkText") + public CommonResult pushDingTalkText(@RequestBody @Valid DevPushDingTalkTextParam devPushDingTalkTextParam) { + devPushService.pushDingTalkText(devPushDingTalkTextParam); + return CommonResult.ok(); + } + + /** + * 推送消息——钉钉MARKDOWN + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 4) + @Operation(summary = "推送钉钉MARKDOWN消息") + @CommonLog("推送钉钉MARKDOWN消息") + @PostMapping("/dev/push/pushDingTalkMarkdown") + public CommonResult pushDingTalkMarkdown(@RequestBody @Valid DevPushDingTalkMarkdownParam devPushDingTalkMarkdownParam) { + devPushService.pushDingTalkMarkdown(devPushDingTalkMarkdownParam); + return CommonResult.ok(); + } + + /** + * 推送消息——钉钉LINK + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 5) + @Operation(summary = "推送钉钉LINK消息") + @CommonLog("推送钉钉LINK消息") + @PostMapping("/dev/push/pushDingTalkLink") + public CommonResult pushDingTalkLink(@RequestBody @Valid DevPushDingTalkLinkParam devPushDingTalkLinkParam) { + devPushService.pushDingTalkLink(devPushDingTalkLinkParam); + return CommonResult.ok(); + } + + /** + * 推送消息——企业微信TXT + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 6) + @Operation(summary = "推送企业微信TEXT消息") + @CommonLog("推送企业微信TEXT消息") + @PostMapping("/dev/push/pushWorkWechatText") + public CommonResult pushWorkWechatText(@RequestBody @Valid DevPushWorkWechatTextParam devPushWorkWechatTextParam) { + devPushService.pushWorkWechatText(devPushWorkWechatTextParam); + return CommonResult.ok(); + } + + /** + * 推送消息——企业微信MARKDOWN + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 7) + @Operation(summary = "推送企业微信MARKDOWN消息") + @CommonLog("推送企业微信MARKDOWN消息") + @PostMapping("/dev/push/pushWorkWechatMarkdown") + public CommonResult pushWorkWechatMarkdown(@RequestBody @Valid DevPushWorkWechatMarkdownParam devPushWorkWechatMarkdownParam) { + devPushService.pushWorkWechatMarkdown(devPushWorkWechatMarkdownParam); + return CommonResult.ok(); + } + + /** + * 推送消息——企业微信NEWS + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + @ApiOperationSupport(order = 8) + @Operation(summary = "推送企业微信NEWS消息") + @CommonLog("推送企业微信NEWS消息") + @PostMapping("/dev/push/pushWorkWechatNews") + public CommonResult pushWorkWechatNews(@RequestBody @Valid DevPushWorkWechatNewsParam devPushWorkWechatNewsParam) { + devPushService.pushWorkWechatNews(devPushWorkWechatNewsParam); + return CommonResult.ok(); + } + + /** + * 获取推送消息分页 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 9) + @Operation(summary = "获取推送消息分页") + @GetMapping("/dev/push/page") + public CommonResult> page(DevPushPageParam devPushPageParam) { + return CommonResult.data(devPushService.page(devPushPageParam)); + } + + /** + * 删除推送消息 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 10) + @Operation(summary = "删除推送消息") + @CommonLog("删除推送消息") + @PostMapping("/dev/push/delete") + public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") + List devPushIdParamList) { + devPushService.delete(devPushIdParamList); + return CommonResult.ok(); + } + + /** + * 获取推送消息详情 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 11) + @Operation(summary = "获取推送消息详情") + @GetMapping("/dev/push/detail") + public CommonResult detail(@Valid DevPushIdParam devPushIdParam) { + return CommonResult.data(devPushService.detail(devPushIdParam)); + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/entity/DevPush.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/entity/DevPush.java new file mode 100644 index 00000000..f24bdf15 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/entity/DevPush.java @@ -0,0 +1,59 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.entity; + +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +/** + * 消息推送实体 + * + * @author xuyuxiang + * @date 2022/2/23 18:27 + **/ +@Getter +@Setter +@TableName("DEV_PUSH") +public class DevPush extends CommonEntity { + + /** id */ + @Schema(description = "主键") + private String id; + + /** 消息引擎 */ + @Schema(description = "消息引擎") + private String engine; + + /** 消息类别 */ + @Schema(description = "消息类别") + private String type; + + /** 消息标题 */ + @Schema(description = "消息标题") + private String title; + + /** 消息内容 */ + @Schema(description = "消息内容") + private String content; + + /** 回执信息 */ + @Schema(description = "回执信息") + private String receiptInfo; + + /** 扩展信息 */ + @Schema(description = "扩展信息") + private String extJson; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushEngineTypeEnum.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushEngineTypeEnum.java new file mode 100644 index 00000000..d11971a2 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushEngineTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.enums; + +import lombok.Getter; + +/** + * 消息推送引擎类型枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum DevPushEngineTypeEnum { + + /** 钉钉 */ + DINGTALK("DINGTALK"), + + /** 飞书 */ + FEISHU("FEISHU"), + + /** 企业微信 */ + WORKWECHAT("WORKWECHAT"); + + private final String value; + + DevPushEngineTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushMessageTypeEnum.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushMessageTypeEnum.java new file mode 100644 index 00000000..21d778e7 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushMessageTypeEnum.java @@ -0,0 +1,43 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.enums; + +import lombok.Getter; + +/** + * 消息推送消息类型枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum DevPushMessageTypeEnum { + + /** 文本 */ + TEXT("TEXT"), + + /** 富文本 */ + MARKDOWN("MARKDOWN"), + + /** 链接 */ + LINK("LINK"), + + /** 文章 */ + NEWS("NEWS"); + + private final String value; + + DevPushMessageTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushNoticeTypeEnum.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushNoticeTypeEnum.java new file mode 100644 index 00000000..c70f4b40 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/enums/DevPushNoticeTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.enums; + +import lombok.Getter; + +/** + * 消息推送通知类型枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum DevPushNoticeTypeEnum { + + /** 无通知 */ + NONE("NONE"), + + /** 通知指定手机号 */ + PHONE("PHONE"), + + /** 通知所有人 */ + ALL("ALL"); + + private final String value; + + DevPushNoticeTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/mapper/DevPushMapper.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/mapper/DevPushMapper.java new file mode 100644 index 00000000..47c5c3a6 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/mapper/DevPushMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.dev.modular.push.entity.DevPush; + +/** + * 消息推送Mapper接口 + * + * @author xuyuxiang + * @date 2022/2/23 18:40 + **/ +public interface DevPushMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/mapper/mapping/DevPushMapper.xml b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/mapper/mapping/DevPushMapper.xml new file mode 100644 index 00000000..573cc7ec --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/mapper/mapping/DevPushMapper.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkLinkParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkLinkParam.java new file mode 100644 index 00000000..d4dfafd4 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkLinkParam.java @@ -0,0 +1,50 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 推送消息——钉钉LINK参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushDingTalkLinkParam { + + /** 消息标题 */ + @Schema(description = "消息标题") + @NotBlank(message = "title不能为空") + private String title; + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; + + /** 图片url */ + @Schema(description = "图片url") + @NotBlank(message = "picUrl不能为空") + private String picUrl; + + /** 跳转地址 */ + @Schema(description = "跳转地址") + @NotBlank(message = "messageUrl不能为空") + private String messageUrl; + +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkMarkdownParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkMarkdownParam.java new file mode 100644 index 00000000..7fba299d --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkMarkdownParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 推送消息——钉钉MARKDOWN参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushDingTalkMarkdownParam { + + /** 消息标题 */ + @Schema(description = "消息标题") + @NotBlank(message = "title不能为空") + private String title; + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; + + /** 通知类型(无、全部) */ + @Schema(description = "通知类型") + @NotBlank(message = "noticeType不能为空") + private String noticeType; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkTextParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkTextParam.java new file mode 100644 index 00000000..a93f00a1 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDingTalkTextParam.java @@ -0,0 +1,43 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 推送消息——钉钉TXT参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushDingTalkTextParam { + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; + + /** 通知类型(无、指定手机、全部) */ + @Schema(description = "通知类型") + @NotBlank(message = "noticeType不能为空") + private String noticeType; + + /** 手机号 */ + @Schema(description = "手机号") + private String phones; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDynamicTextParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDynamicTextParam.java new file mode 100644 index 00000000..15dff49e --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushDynamicTextParam.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import jakarta.validation.constraints.NotNull; +import lombok.Getter; +import lombok.Setter; + +/** + * 动态推送消息参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushDynamicTextParam { + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; + + /** 是否@所有人 */ + @Schema(description = "是否@所有人") + @NotNull(message = "noticeAll不能为空") + private Boolean noticeAll = false; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushFeiShuTextParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushFeiShuTextParam.java new file mode 100644 index 00000000..4e10ff2f --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushFeiShuTextParam.java @@ -0,0 +1,39 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 推送消息——飞书TXT参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushFeiShuTextParam { + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; + + /** 通知类型(无、全部) */ + @Schema(description = "通知类型") + @NotBlank(message = "noticeType不能为空") + private String noticeType; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushIdParam.java new file mode 100644 index 00000000..3dc16d29 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushIdParam.java @@ -0,0 +1,34 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 消息推送Id参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushIdParam { + + /** id */ + @Schema(description = "id") + @NotBlank(message = "id不能为空") + private String id; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushPageParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushPageParam.java new file mode 100644 index 00000000..e53aaa6c --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushPageParam.java @@ -0,0 +1,52 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; + +/** + * 消息推送查询参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushPageParam { + + /** 当前页 */ + @Schema(description = "当前页码") + private Integer current; + + /** 每页条数 */ + @Schema(description = "每页条数") + private Integer size; + + /** 排序字段 */ + @Schema(description = "排序字段,字段驼峰名称,如:userName") + private String sortField; + + /** 排序方式 */ + @Schema(description = "排序方式,升序:ASCEND;降序:DESCEND") + private String sortOrder; + + /** 消息引擎 */ + @Schema(description = "消息引擎") + private String engine; + + /** 消息标题关键词 */ + @Schema(description = "消息标题关键词") + private String searchKey; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatMarkdownParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatMarkdownParam.java new file mode 100644 index 00000000..cabe5f54 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatMarkdownParam.java @@ -0,0 +1,39 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 推送消息——企业微信MARKDOWN参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushWorkWechatMarkdownParam { + + /** 消息标题 */ + @Schema(description = "消息标题") + @NotBlank(message = "title不能为空") + private String title; + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatNewsParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatNewsParam.java new file mode 100644 index 00000000..d7646973 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatNewsParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 推送消息——企业微信NEWS参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushWorkWechatNewsParam { + + /** 消息标题 */ + @Schema(description = "消息标题") + @NotBlank(message = "title不能为空") + private String title; + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; + + /** 图片url */ + @Schema(description = "图片url") + @NotBlank(message = "picUrl不能为空") + private String picUrl; + + /** 跳转地址 */ + @Schema(description = "跳转地址") + @NotBlank(message = "messageUrl不能为空") + private String messageUrl; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatTextParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatTextParam.java new file mode 100644 index 00000000..37ea2d1a --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/param/DevPushWorkWechatTextParam.java @@ -0,0 +1,43 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 推送消息——企业微信TXT参数 + * + * @author xuyuxiang + * @date 2022/6/21 15:38 + **/ +@Getter +@Setter +public class DevPushWorkWechatTextParam { + + /** 消息内容 */ + @Schema(description = "消息内容") + @NotBlank(message = "content不能为空") + private String content; + + /** 通知类型(无、指定手机、全部) */ + @Schema(description = "通知类型") + @NotBlank(message = "noticeType不能为空") + private String noticeType; + + /** 手机号 */ + @Schema(description = "手机号") + private String phones; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/provider/DevPushApiProvider.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/provider/DevPushApiProvider.java new file mode 100644 index 00000000..acf0b97a --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/provider/DevPushApiProvider.java @@ -0,0 +1,127 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.provider; + +import cn.hutool.core.util.ObjectUtil; +import jakarta.annotation.Resource; +import org.springframework.stereotype.Service; +import vip.xiaonuo.dev.api.DevPushApi; +import vip.xiaonuo.dev.modular.push.enums.DevPushNoticeTypeEnum; +import vip.xiaonuo.dev.modular.push.param.*; +import vip.xiaonuo.dev.modular.push.service.DevPushService; + +/** + * 消息推送API接口提供者 + * + * @author xuyuxiang + * @date 2022/6/22 15:32 + **/ +@Service +public class DevPushApiProvider implements DevPushApi { + + @Resource + private DevPushService devPushService; + + @Override + public void pushDynamicText(String content, boolean noticeAll) { + DevPushDynamicTextParam dynamicTextParam = new DevPushDynamicTextParam(); + dynamicTextParam.setContent(content); + dynamicTextParam.setNoticeAll(noticeAll); + devPushService.pushDynamicText(dynamicTextParam); + } + + @Override + public void pushFeiShuText(String content, boolean noticeAll) { + DevPushFeiShuTextParam pushFeiShuTextParam = new DevPushFeiShuTextParam(); + pushFeiShuTextParam.setContent(content); + if(noticeAll){ + pushFeiShuTextParam.setNoticeType(DevPushNoticeTypeEnum.ALL.getValue()); + } else { + pushFeiShuTextParam.setNoticeType(DevPushNoticeTypeEnum.NONE.getValue()); + } + devPushService.pushFeiShuText(pushFeiShuTextParam); + } + + @Override + public void pushDingTalkText(String content, boolean noticeAll, String phones) { + DevPushDingTalkTextParam pushDingTalkTextParam = new DevPushDingTalkTextParam(); + pushDingTalkTextParam.setContent(content); + if(noticeAll){ + pushDingTalkTextParam.setNoticeType(DevPushNoticeTypeEnum.ALL.getValue()); + } else { + if(ObjectUtil.isNotEmpty(phones)){ + pushDingTalkTextParam.setNoticeType(DevPushNoticeTypeEnum.PHONE.getValue()); + } else { + pushDingTalkTextParam.setNoticeType(DevPushNoticeTypeEnum.NONE.getValue()); + } + } + devPushService.pushDingTalkText(pushDingTalkTextParam); + } + + @Override + public void pushDingTalkMarkdown(String title, String content, boolean noticeAll) { + DevPushDingTalkMarkdownParam devPushDingTalkMarkdownParam = new DevPushDingTalkMarkdownParam(); + devPushDingTalkMarkdownParam.setTitle(title); + devPushDingTalkMarkdownParam.setContent(content); + if(noticeAll){ + devPushDingTalkMarkdownParam.setNoticeType(DevPushNoticeTypeEnum.ALL.getValue()); + } else { + devPushDingTalkMarkdownParam.setNoticeType(DevPushNoticeTypeEnum.NONE.getValue()); + } + devPushService.pushDingTalkMarkdown(devPushDingTalkMarkdownParam); + } + + @Override + public void pushDingTalkLink(String title, String content, String picUrl, String messageUrl) { + DevPushDingTalkLinkParam devPushDingTalkLinkParam = new DevPushDingTalkLinkParam(); + devPushDingTalkLinkParam.setTitle(title); + devPushDingTalkLinkParam.setContent(content); + devPushDingTalkLinkParam.setPicUrl(picUrl); + devPushDingTalkLinkParam.setMessageUrl(messageUrl); + devPushService.pushDingTalkLink(devPushDingTalkLinkParam); + } + + @Override + public void pushWorkWechatText(String content, boolean noticeAll, String phones) { + DevPushWorkWechatTextParam devPushWorkWechatTextParam = new DevPushWorkWechatTextParam(); + devPushWorkWechatTextParam.setContent(content); + if(noticeAll){ + devPushWorkWechatTextParam.setNoticeType(DevPushNoticeTypeEnum.ALL.getValue()); + } else { + if(ObjectUtil.isNotEmpty(phones)){ + devPushWorkWechatTextParam.setNoticeType(DevPushNoticeTypeEnum.PHONE.getValue()); + } else { + devPushWorkWechatTextParam.setNoticeType(DevPushNoticeTypeEnum.NONE.getValue()); + } + } + devPushService.pushWorkWechatText(devPushWorkWechatTextParam); + } + + @Override + public void pushWorkWechatMarkdown(String title, String content) { + DevPushWorkWechatMarkdownParam devPushWorkWechatMarkdownParam = new DevPushWorkWechatMarkdownParam(); + devPushWorkWechatMarkdownParam.setTitle(title); + devPushWorkWechatMarkdownParam.setContent(content); + devPushService.pushWorkWechatMarkdown(devPushWorkWechatMarkdownParam); + } + + @Override + public void pushWorkWechatNews(String title, String content, String picUrl, String messageUrl) { + DevPushWorkWechatNewsParam devPushWorkWechatNewsParam = new DevPushWorkWechatNewsParam(); + devPushWorkWechatNewsParam.setTitle(title); + devPushWorkWechatNewsParam.setContent(content); + devPushWorkWechatNewsParam.setPicUrl(picUrl); + devPushWorkWechatNewsParam.setMessageUrl(messageUrl); + devPushService.pushWorkWechatNews(devPushWorkWechatNewsParam); + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/service/DevPushService.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/service/DevPushService.java new file mode 100644 index 00000000..1ab95148 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/service/DevPushService.java @@ -0,0 +1,137 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.service; + +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.dev.modular.push.entity.DevPush; +import vip.xiaonuo.dev.modular.push.param.*; + +import java.util.List; + +/** + * 消息推送Service接口 + * + * @author xuyuxiang + * @date 2022/2/23 18:27 + **/ +public interface DevPushService extends IService { + + /** + * 动态推送消息 + * + * @param engine 推送引擎 + * @param content 文本内容 + * @param noticeAll 是否@所有人 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushDynamicText(String engine, String content, boolean noticeAll); + + /** + * 动态推送消息 + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushDynamicText(DevPushDynamicTextParam devPushDynamicTextParam); + + /** + * 推送消息——飞书TXT + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushFeiShuText(DevPushFeiShuTextParam devPushFeiShuTextParam); + + /** + * 推送消息——钉钉TXT + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushDingTalkText(DevPushDingTalkTextParam devPushDingTalkTextParam); + + /** + * 推送消息——钉钉MARKDOWN + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushDingTalkMarkdown(DevPushDingTalkMarkdownParam devPushDingTalkMarkdownParam); + + /** + * 推送消息——钉钉LINK + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushDingTalkLink(DevPushDingTalkLinkParam devPushDingTalkLinkParam); + + /** + * 推送消息——企业微信TXT + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushWorkWechatText(DevPushWorkWechatTextParam devPushWorkWechatTextParam); + + /** + * 推送消息——企业微信MARKDOWN + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushWorkWechatMarkdown(DevPushWorkWechatMarkdownParam devPushWorkWechatMarkdownParam); + + /** + * 推送消息——企业微信NEWS + * + * @author xuyuxiang + * @date 2022/4/24 20:47 + */ + void pushWorkWechatNews(DevPushWorkWechatNewsParam devPushWorkWechatNewsParam); + + /** + * 获取推送消息分页 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + Page page(DevPushPageParam devPushPageParam); + + /** + * 删除推送消息 + * + * @author xuyuxiang + * @date 2022/8/4 10:36 + **/ + void delete(List devPushIdParamList); + + /** + * 获取推送消息详情 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + DevPush detail(DevPushIdParam devPushIdParam); + + /** + * 获取推送消息详情 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + DevPush queryEntity(String id); +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/service/impl/DevPushServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/service/impl/DevPushServiceImpl.java new file mode 100644 index 00000000..957193a2 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/service/impl/DevPushServiceImpl.java @@ -0,0 +1,271 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.service.impl; + +import cn.hutool.core.collection.CollStreamUtil; +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import jakarta.annotation.Resource; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import vip.xiaonuo.common.enums.CommonSortOrderEnum; +import vip.xiaonuo.common.exception.CommonException; +import vip.xiaonuo.common.page.CommonPageRequest; +import vip.xiaonuo.dev.api.DevConfigApi; +import vip.xiaonuo.dev.modular.push.entity.DevPush; +import vip.xiaonuo.dev.modular.push.enums.DevPushEngineTypeEnum; +import vip.xiaonuo.dev.modular.push.enums.DevPushMessageTypeEnum; +import vip.xiaonuo.dev.modular.push.enums.DevPushNoticeTypeEnum; +import vip.xiaonuo.dev.modular.push.mapper.DevPushMapper; +import vip.xiaonuo.dev.modular.push.param.*; +import vip.xiaonuo.dev.modular.push.service.DevPushService; +import vip.xiaonuo.dev.modular.push.util.DevPushDingTalkUtil; +import vip.xiaonuo.dev.modular.push.util.DevPushFeiShuUtil; +import vip.xiaonuo.dev.modular.push.util.DevPushWorkWechatUtil; + +import java.util.List; + +/** + * 邮件Service接口实现类 + * + * @author xuyuxiang + * @date 2022/2/23 18:43 + **/ +@Service +public class DevPushServiceImpl extends ServiceImpl implements DevPushService { + + /** 默认消息引擎 */ + private static final String SNOWY_SYS_DEFAULT_PUSH_ENGINE_KEY = "SNOWY_SYS_DEFAULT_PUSH_ENGINE"; + + @Resource + private DevConfigApi devConfigApi; + + @Override + public void pushDynamicText(String engine, String content, boolean noticeAll) { + if(engine.equals(DevPushEngineTypeEnum.DINGTALK.getValue())) { + DevPushDingTalkTextParam devPushDingTalkTextParam = new DevPushDingTalkTextParam(); + devPushDingTalkTextParam.setContent(content); + devPushDingTalkTextParam.setNoticeType(noticeAll?DevPushNoticeTypeEnum.ALL.getValue(): + DevPushNoticeTypeEnum.NONE.getValue()); + this.pushDingTalkText(devPushDingTalkTextParam); + } else if (engine.equals(DevPushEngineTypeEnum.FEISHU.getValue())) { + DevPushFeiShuTextParam devPushFeiShuTextParam = new DevPushFeiShuTextParam(); + devPushFeiShuTextParam.setContent(content); + devPushFeiShuTextParam.setNoticeType(noticeAll?DevPushNoticeTypeEnum.ALL.getValue(): + DevPushNoticeTypeEnum.NONE.getValue()); + this.pushFeiShuText(devPushFeiShuTextParam); + } else if (engine.equals(DevPushEngineTypeEnum.WORKWECHAT.getValue())) { + DevPushWorkWechatTextParam devPushWorkWechatTextParam = new DevPushWorkWechatTextParam(); + devPushWorkWechatTextParam.setContent(content); + devPushWorkWechatTextParam.setNoticeType(noticeAll?DevPushNoticeTypeEnum.ALL.getValue(): + DevPushNoticeTypeEnum.NONE.getValue()); + this.pushWorkWechatText(devPushWorkWechatTextParam); + } else { + throw new CommonException("不支持的消息推送引擎:{}", engine); + } + } + + @Override + public void pushDynamicText(DevPushDynamicTextParam devPushDynamicTextParam) { + String defaultEmailEngine = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PUSH_ENGINE_KEY); + if(ObjectUtil.isEmpty(defaultEmailEngine)) { + throw new CommonException("请联系管理员配置默认消息推送引擎"); + } + String content = devPushDynamicTextParam.getContent(); + Boolean noticeAll = devPushDynamicTextParam.getNoticeAll(); + this.pushDynamicText(defaultEmailEngine, content, noticeAll); + } + + @Override + public void pushFeiShuText(DevPushFeiShuTextParam devPushFeiShuTextParam) { + String noticeType = devPushFeiShuTextParam.getNoticeType(); + String receiptInfo; + if(DevPushNoticeTypeEnum.NONE.getValue().equals(noticeType)){ + receiptInfo = DevPushFeiShuUtil.pushFeiShuText(devPushFeiShuTextParam.getContent(), false); + } else if(DevPushNoticeTypeEnum.ALL.getValue().equals(noticeType)) { + receiptInfo = DevPushFeiShuUtil.pushFeiShuText(devPushFeiShuTextParam.getContent(), true); + } else { + throw new CommonException("不支持的通知类型:{}", noticeType); + } + DevPush devPush = new DevPush(); + devPush.setEngine(DevPushEngineTypeEnum.FEISHU.getValue()); + devPush.setType(DevPushMessageTypeEnum.TEXT.getValue()); + devPush.setTitle(devPushFeiShuTextParam.getContent()); + devPush.setContent(devPushFeiShuTextParam.getContent()); + devPush.setReceiptInfo(receiptInfo); + this.save(devPush); + } + + @Override + public void pushDingTalkText(DevPushDingTalkTextParam devPushDingTalkTextParam) { + String noticeType = devPushDingTalkTextParam.getNoticeType(); + String receiptInfo; + if(DevPushNoticeTypeEnum.NONE.getValue().equals(noticeType)){ + receiptInfo = DevPushDingTalkUtil.pushDingTalkText(devPushDingTalkTextParam.getContent(), + false, null); + } else if(DevPushNoticeTypeEnum.ALL.getValue().equals(noticeType)) { + receiptInfo = DevPushDingTalkUtil.pushDingTalkText(devPushDingTalkTextParam.getContent(), + true, null); + } else if(DevPushNoticeTypeEnum.PHONE.getValue().equals(noticeType)) { + if(ObjectUtil.isEmpty(devPushDingTalkTextParam.getPhones())) { + throw new CommonException("手机号不能为空"); + } else { + receiptInfo = DevPushDingTalkUtil.pushDingTalkText(devPushDingTalkTextParam.getContent(), + false, devPushDingTalkTextParam.getPhones()); + } + } else { + throw new CommonException("不支持的通知类型:{}", noticeType); + } + DevPush devPush = new DevPush(); + devPush.setEngine(DevPushEngineTypeEnum.DINGTALK.getValue()); + devPush.setType(DevPushMessageTypeEnum.TEXT.getValue()); + devPush.setTitle(devPushDingTalkTextParam.getContent()); + devPush.setContent(devPushDingTalkTextParam.getContent()); + devPush.setReceiptInfo(receiptInfo); + this.save(devPush); + } + + @Override + public void pushDingTalkMarkdown(DevPushDingTalkMarkdownParam devPushDingTalkMarkdownParam) { + String noticeType = devPushDingTalkMarkdownParam.getNoticeType(); + String receiptInfo; + if(DevPushNoticeTypeEnum.NONE.getValue().equals(noticeType)){ + receiptInfo = DevPushDingTalkUtil.pushDingTalkMarkdown(devPushDingTalkMarkdownParam.getTitle(), + devPushDingTalkMarkdownParam.getContent(), false); + } else if(DevPushNoticeTypeEnum.ALL.getValue().equals(noticeType)) { + receiptInfo = DevPushDingTalkUtil.pushDingTalkMarkdown(devPushDingTalkMarkdownParam.getTitle(), + devPushDingTalkMarkdownParam.getContent(), true); + } else { + throw new CommonException("不支持的通知类型:{}", noticeType); + } + DevPush devPush = new DevPush(); + devPush.setEngine(DevPushEngineTypeEnum.DINGTALK.getValue()); + devPush.setType(DevPushMessageTypeEnum.MARKDOWN.getValue()); + devPush.setTitle(devPushDingTalkMarkdownParam.getTitle()); + devPush.setContent(devPushDingTalkMarkdownParam.getContent()); + devPush.setReceiptInfo(receiptInfo); + this.save(devPush); + } + + @Override + public void pushDingTalkLink(DevPushDingTalkLinkParam devPushDingTalkLinkParam) { + String receiptInfo = DevPushDingTalkUtil.pushDingTalkLink(devPushDingTalkLinkParam.getTitle(), + devPushDingTalkLinkParam.getContent(), + devPushDingTalkLinkParam.getPicUrl(), + devPushDingTalkLinkParam.getMessageUrl()); + DevPush devPush = new DevPush(); + devPush.setEngine(DevPushEngineTypeEnum.DINGTALK.getValue()); + devPush.setType(DevPushMessageTypeEnum.LINK.getValue()); + devPush.setTitle(devPushDingTalkLinkParam.getTitle()); + devPush.setContent(devPushDingTalkLinkParam.getContent()); + devPush.setReceiptInfo(receiptInfo); + this.save(devPush); + } + + @Override + public void pushWorkWechatText(DevPushWorkWechatTextParam devPushWorkWechatTextParam) { + String noticeType = devPushWorkWechatTextParam.getNoticeType(); + String receiptInfo; + if(DevPushNoticeTypeEnum.NONE.getValue().equals(noticeType)){ + receiptInfo = DevPushWorkWechatUtil.pushWorkWechatText(devPushWorkWechatTextParam.getContent(), + false, null); + } else if(DevPushNoticeTypeEnum.ALL.getValue().equals(noticeType)) { + receiptInfo = DevPushWorkWechatUtil.pushWorkWechatText(devPushWorkWechatTextParam.getContent(), + true, null); + } else if(DevPushNoticeTypeEnum.PHONE.getValue().equals(noticeType)) { + if(ObjectUtil.isEmpty(devPushWorkWechatTextParam.getPhones())) { + throw new CommonException("手机号不能为空"); + } else { + receiptInfo = DevPushWorkWechatUtil.pushWorkWechatText(devPushWorkWechatTextParam.getContent(), + false, devPushWorkWechatTextParam.getPhones()); + } + } else { + throw new CommonException("不支持的通知类型:{}", noticeType); + } + DevPush devPush = new DevPush(); + devPush.setEngine(DevPushEngineTypeEnum.WORKWECHAT.getValue()); + devPush.setType(DevPushMessageTypeEnum.TEXT.getValue()); + devPush.setTitle(devPushWorkWechatTextParam.getContent()); + devPush.setContent(devPushWorkWechatTextParam.getContent()); + devPush.setReceiptInfo(receiptInfo); + this.save(devPush); + } + + @Override + public void pushWorkWechatMarkdown(DevPushWorkWechatMarkdownParam devPushWorkWechatMarkdownParam) { + String receiptInfo = DevPushWorkWechatUtil.pushWorkWechatMarkdown(devPushWorkWechatMarkdownParam.getTitle(), + devPushWorkWechatMarkdownParam.getContent()); + DevPush devPush = new DevPush(); + devPush.setEngine(DevPushEngineTypeEnum.WORKWECHAT.getValue()); + devPush.setType(DevPushMessageTypeEnum.MARKDOWN.getValue()); + devPush.setTitle(devPushWorkWechatMarkdownParam.getTitle()); + devPush.setContent(devPushWorkWechatMarkdownParam.getContent()); + devPush.setReceiptInfo(receiptInfo); + this.save(devPush); + } + + @Override + public void pushWorkWechatNews(DevPushWorkWechatNewsParam devPushWorkWechatNewsParam) { + String receiptInfo = DevPushWorkWechatUtil.pushWorkWechatNews(devPushWorkWechatNewsParam.getTitle(), + devPushWorkWechatNewsParam.getContent(), + devPushWorkWechatNewsParam.getPicUrl(), + devPushWorkWechatNewsParam.getMessageUrl()); + DevPush devPush = new DevPush(); + devPush.setEngine(DevPushEngineTypeEnum.WORKWECHAT.getValue()); + devPush.setType(DevPushMessageTypeEnum.NEWS.getValue()); + devPush.setTitle(devPushWorkWechatNewsParam.getTitle()); + devPush.setContent(devPushWorkWechatNewsParam.getContent()); + devPush.setReceiptInfo(receiptInfo); + this.save(devPush); + } + + @Override + public Page page(DevPushPageParam devPushPageParam) { + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + if(ObjectUtil.isNotEmpty(devPushPageParam.getEngine())) { + queryWrapper.lambda().eq(DevPush::getEngine, devPushPageParam.getEngine()); + } + if(ObjectUtil.isNotEmpty(devPushPageParam.getSearchKey())) { + queryWrapper.lambda().like(DevPush::getTitle, devPushPageParam.getSearchKey()); + } + if(ObjectUtil.isAllNotEmpty(devPushPageParam.getSortField(), devPushPageParam.getSortOrder())) { + CommonSortOrderEnum.validate(devPushPageParam.getSortOrder()); + queryWrapper.orderBy(true, devPushPageParam.getSortOrder().equals(CommonSortOrderEnum.ASC.getValue()), + StrUtil.toUnderlineCase(devPushPageParam.getSortField())); + } + return this.page(CommonPageRequest.defaultPage(), queryWrapper); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void delete(List devPushIdParamList) { + this.removeByIds(CollStreamUtil.toList(devPushIdParamList, DevPushIdParam::getId)); + } + + @Override + public DevPush detail(DevPushIdParam devPushIdParam) { + return this.queryEntity(devPushIdParam.getId()); + } + + @Override + public DevPush queryEntity(String id) { + DevPush devPush = this.getById(id); + if(ObjectUtil.isEmpty(devPush)) { + throw new CommonException("消息推送记录不存在,id值为:{}", id); + } + return devPush; + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushDingTalkUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushDingTalkUtil.java new file mode 100644 index 00000000..a16e52d3 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushDingTalkUtil.java @@ -0,0 +1,170 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.util; + +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONUtil; +import lombok.extern.slf4j.Slf4j; +import org.dromara.oa.api.OaSender; +import org.dromara.oa.comm.entity.Request; +import org.dromara.oa.comm.entity.Response; +import org.dromara.oa.comm.enums.MessageType; +import org.dromara.oa.core.dingTalk.config.DingTalkConfig; +import org.dromara.oa.core.dingTalk.config.DingTalkFactory; +import org.dromara.oa.core.provider.factory.OaFactory; +import org.dromara.oa.core.provider.factory.ProviderFactoryHolder; +import vip.xiaonuo.common.exception.CommonException; +import vip.xiaonuo.dev.api.DevConfigApi; + +/** + * 钉钉消息推送工具类 + * + * @author xuyuxiang + * @date 2022/1/2 17:05 + */ +@Slf4j +public class DevPushDingTalkUtil { + + private static OaSender oaSender; + + private static final String SNOWY_PUSH_DINGTALK_SIGN_KEY = "SNOWY_PUSH_DINGTALK_SIGN"; + private static final String SNOWY_PUSH_DINGTALK_TOKEN_ID_KEY = "SNOWY_PUSH_DINGTALK_TOKEN_ID"; + + static { + ProviderFactoryHolder.registerFactory(DingTalkFactory.instance()); + } + + /** + * 初始化操作的客户端 + * + * @author xuyuxiang + * @date 2022/1/5 23:24 + */ + private static void initClient() { + + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + + /* sign */ + String sign = devConfigApi.getValueByKey(SNOWY_PUSH_DINGTALK_SIGN_KEY); + + if(ObjectUtil.isEmpty(sign)) { + throw new CommonException("钉钉推送操作客户端未正确配置:sign为空"); + } + + /* tokenId */ + String tokenId = devConfigApi.getValueByKey(SNOWY_PUSH_DINGTALK_TOKEN_ID_KEY); + + if(ObjectUtil.isEmpty(tokenId)) { + throw new CommonException("钉钉推送操作客户端未正确配置:tokenId为空"); + } + + DingTalkConfig dingTalkConfig = new DingTalkConfig(); + dingTalkConfig.setConfigId(tokenId); + dingTalkConfig.setSign(sign); + dingTalkConfig.setTokenId(tokenId); + OaFactory.createAndRegisterOaSender(dingTalkConfig); + oaSender = OaFactory.getSmsOaBlend(tokenId); + } + + /** + * 发送文本消息 + * + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @param phones 通知的用户手机号,英文逗号分割 + * @return 发送成功的回执信息 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + public static String pushDingTalkText(String content, boolean noticeAll, String phones) { + try { + initClient(); + Request request = new Request(); + request.setContent(content); + if(noticeAll) { + request.setIsNoticeAll(true); + } else { + if(ObjectUtil.isNotEmpty(phones)) { + request.setPhoneList((StrUtil.split(phones, StrUtil.COMMA))); + } + } + Response response = oaSender.sender(request, MessageType.DING_TALK_TEXT); + if(!response.isSuccess()) { + throw new CommonException("消息推送错误,返回内容:{}", JSONUtil.toJsonStr(response.getData())); + } + return JSONUtil.toJsonStr(response.getData()); + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } + + /** + * 发送MarkDown消息 + * + * @param title 标题 + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @return 发送成功的回执信息 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + public static String pushDingTalkMarkdown(String title, String content, boolean noticeAll) { + try { + initClient(); + Request request = new Request(); + request.setTitle(title); + request.setContent(content); + if(noticeAll) { + request.setIsNoticeAll(true); + } + Response response = oaSender.sender(request, MessageType.DING_TALK_MARKDOWN); + if(!response.isSuccess()) { + throw new CommonException("消息推送错误,返回内容:{}", JSONUtil.toJsonStr(response.getData())); + } + return JSONUtil.toJsonStr(response.getData()); + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } + + /** + * 发送链接消息 + * + * @param title 标题 + * @param content 内容 + * @param picUrl 封面图片地址 + * @param messageUrl 消息链接地址 + * @return 发送成功的回执信息 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + public static String pushDingTalkLink(String title, String content, String picUrl,String messageUrl) { + try { + initClient(); + Request request = new Request(); + request.setTitle(title); + request.setContent(content); + request.setPicUrl(picUrl); + request.setMessageUrl(messageUrl); + Response response = oaSender.sender(request, MessageType.DING_TALK_LINK); + if(!response.isSuccess()) { + throw new CommonException("消息推送错误,返回内容:{}", JSONUtil.toJsonStr(response.getData())); + } + return JSONUtil.toJsonStr(response.getData()); + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushFeiShuUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushFeiShuUtil.java new file mode 100644 index 00000000..f05b56e1 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushFeiShuUtil.java @@ -0,0 +1,97 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.util; + +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONUtil; +import lombok.extern.slf4j.Slf4j; +import org.dromara.oa.api.OaSender; +import org.dromara.oa.comm.entity.Request; +import org.dromara.oa.comm.entity.Response; +import org.dromara.oa.comm.enums.MessageType; +import org.dromara.oa.core.byteTalk.config.ByteTalkConfig; +import org.dromara.oa.core.byteTalk.config.ByteTalkFactory; +import org.dromara.oa.core.provider.factory.OaFactory; +import org.dromara.oa.core.provider.factory.ProviderFactoryHolder; +import vip.xiaonuo.common.exception.CommonException; +import vip.xiaonuo.dev.api.DevConfigApi; + +/** + * 飞书消息推送工具类 + * + * @author xuyuxiang + * @date 2022/1/2 17:05 + */ +@Slf4j +public class DevPushFeiShuUtil { + + private static OaSender oaSender; + + private static final String SNOWY_PUSH_FEISHU_TOKEN_ID_KEY = "SNOWY_PUSH_FEISHU_TOKEN_ID"; + + static { + ProviderFactoryHolder.registerFactory(ByteTalkFactory.instance()); + } + + /** + * 初始化操作的客户端 + * + * @author xuyuxiang + * @date 2022/1/5 23:24 + */ + private static void initClient() { + + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + + /* tokenId */ + String tokenId = devConfigApi.getValueByKey(SNOWY_PUSH_FEISHU_TOKEN_ID_KEY); + + if(ObjectUtil.isEmpty(tokenId)) { + throw new CommonException("飞书推送操作客户端未正确配置:tokenId为空"); + } + + ByteTalkConfig byteTalkConfig = new ByteTalkConfig(); + byteTalkConfig.setConfigId(tokenId); + byteTalkConfig.setTokenId(tokenId); + OaFactory.createAndRegisterOaSender(byteTalkConfig); + oaSender = OaFactory.getSmsOaBlend(tokenId); + } + + /** + * 发送文本消息 + * + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @return 发送成功的回执信息 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + public static String pushFeiShuText(String content, boolean noticeAll) { + try { + initClient(); + Request request = new Request(); + request.setContent(content); + if(noticeAll) { + request.setIsNoticeAll(true); + } + Response response = oaSender.sender(request, MessageType.BYTE_TALK_TEXT); + if(!response.isSuccess()) { + throw new CommonException("消息推送错误,返回内容:{}", JSONUtil.toJsonStr(response.getData())); + } + return JSONUtil.toJsonStr(response.getData()); + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushWorkWechatUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushWorkWechatUtil.java new file mode 100644 index 00000000..837c058d --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/push/util/DevPushWorkWechatUtil.java @@ -0,0 +1,161 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.push.util; + +import cn.hutool.core.collection.CollectionUtil; +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONUtil; +import lombok.extern.slf4j.Slf4j; +import org.dromara.oa.api.OaSender; +import org.dromara.oa.comm.entity.Request; +import org.dromara.oa.comm.entity.Response; +import org.dromara.oa.comm.entity.WeTalkRequestArticle; +import org.dromara.oa.comm.enums.MessageType; +import org.dromara.oa.core.provider.factory.OaFactory; +import org.dromara.oa.core.provider.factory.ProviderFactoryHolder; +import org.dromara.oa.core.weTalk.config.WeTalkConfig; +import org.dromara.oa.core.weTalk.config.WeTalkFactory; +import vip.xiaonuo.common.exception.CommonException; +import vip.xiaonuo.dev.api.DevConfigApi; + +/** + * 企业微信消息推送工具类 + * + * @author xuyuxiang + * @date 2022/1/2 17:05 + */ +@Slf4j +public class DevPushWorkWechatUtil { + + private static OaSender oaSender; + + private static final String SNOWY_PUSH_WORKWECHAT_TOKEN_ID_KEY = "SNOWY_PUSH_WORKWECHAT_TOKEN_ID"; + + static { + ProviderFactoryHolder.registerFactory(WeTalkFactory.instance()); + } + + /** + * 初始化操作的客户端 + * + * @author xuyuxiang + * @date 2022/1/5 23:24 + */ + private static void initClient() { + + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + + /* tokenId */ + String tokenId = devConfigApi.getValueByKey(SNOWY_PUSH_WORKWECHAT_TOKEN_ID_KEY); + + if(ObjectUtil.isEmpty(tokenId)) { + throw new CommonException("企业微信推送操作客户端未正确配置:tokenId为空"); + } + + WeTalkConfig weTalkConfig = new WeTalkConfig(); + weTalkConfig.setConfigId(tokenId); + weTalkConfig.setTokenId(tokenId); + OaFactory.createAndRegisterOaSender(weTalkConfig); + oaSender = OaFactory.getSmsOaBlend(tokenId); + } + + /** + * 发送文本消息 + * + * @param content 内容 + * @param noticeAll 是否通知所有人 + * @param phones 通知的用户手机号,英文逗号分割 + * @return 发送成功的回执信息 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + public static String pushWorkWechatText(String content, boolean noticeAll, String phones) { + try { + initClient(); + Request request = new Request(); + request.setContent(content); + if(noticeAll) { + request.setIsNoticeAll(true); + } else { + if(ObjectUtil.isNotEmpty(phones)) { + request.setPhoneList((StrUtil.split(phones, StrUtil.COMMA))); + } + } + Response response = oaSender.sender(request, MessageType.WE_TALK_TEXT); + if(!response.isSuccess()) { + throw new CommonException("消息推送错误,返回内容:{}", JSONUtil.toJsonStr(response.getData())); + } + return JSONUtil.toJsonStr(response.getData()); + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } + + /** + * 发送MarkDown消息 + * + * @param title 标题 + * @param content 内容 + * @return 发送成功的回执信息 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + public static String pushWorkWechatMarkdown(String title, String content) { + try { + initClient(); + Request request = new Request(); + request.setTitle(title); + request.setContent(content); + Response response = oaSender.sender(request, MessageType.WE_TALK_MARKDOWN); + if(!response.isSuccess()) { + throw new CommonException("消息推送错误,返回内容:{}", JSONUtil.toJsonStr(response.getData())); + } + return JSONUtil.toJsonStr(response.getData()); + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } + + /** + * 发送新闻消息 + * + * @param title 标题 + * @param content 内容 + * @param picUrl 封面图片地址 + * @param messageUrl 消息链接地址 + * @return 发送成功的回执信息 + * @author xuyuxiang + * @date 2022/2/23 14:24 + */ + public static String pushWorkWechatNews(String title, String content, String picUrl,String messageUrl) { + try { + initClient(); + Request request = new Request(); + WeTalkRequestArticle weTalkRequestArticle = new WeTalkRequestArticle(); + weTalkRequestArticle.setTitle(title); + weTalkRequestArticle.setDescription(content); + weTalkRequestArticle.setPicUrl(picUrl); + weTalkRequestArticle.setUrl(messageUrl); + request.setArticleList(CollectionUtil.newArrayList()); + Response response = oaSender.sender(request, MessageType.WE_TALK_NEWS); + if(!response.isSuccess()) { + throw new CommonException("消息推送错误,返回内容:{}", JSONUtil.toJsonStr(response.getData())); + } + return JSONUtil.toJsonStr(response.getData()); + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/relation/service/impl/DevRelationServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/relation/service/impl/DevRelationServiceImpl.java index 709ed653..891c3f14 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/relation/service/impl/DevRelationServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/relation/service/impl/DevRelationServiceImpl.java @@ -72,41 +72,49 @@ public class DevRelationServiceImpl extends ServiceImpl targetIdList, String category) { this.saveRelationBatch(objectId, targetIdList, category, null, false); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithAppend(String objectId, List targetIdList, String category, List extJsonList) { this.saveRelationBatch(objectId, targetIdList, category, extJsonList, false); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationWithClear(String objectId, String targetId, String category) { this.saveRelation(objectId, targetId, category, null, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationWithClear(String objectId, String targetId, String category, String extJson) { this.saveRelation(objectId, targetId, category, extJson, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithClear(String objectId, List targetIdList, String category) { this.saveRelationBatch(objectId, targetIdList, category, null, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithClear(String objectId, List targetIdList, String category, List extJsonList) { this.saveRelationBatch(objectId, targetIdList, category, extJsonList, true); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/slideshow/service/impl/DevSlideshowServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/slideshow/service/impl/DevSlideshowServiceImpl.java index 596ec324..da146648 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/slideshow/service/impl/DevSlideshowServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/slideshow/service/impl/DevSlideshowServiceImpl.java @@ -147,7 +147,7 @@ public class DevSlideshowServiceImpl extends ServiceImpl sendDynamic(@RequestBody @Valid DevSmsSendDynamicParam devSmsSendDynamicParam) { + devSmsService.sendDynamic(devSmsSendDynamicParam); + return CommonResult.ok(); + } + /** * 发送短信——阿里云 * * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "发送阿里云短信") @CommonLog("发送阿里云短信") @PostMapping("/dev/sms/sendAliyun") @@ -65,6 +84,7 @@ public class DevSmsController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "发送腾讯云短信") @CommonLog("发送腾讯云短信") @PostMapping("/dev/sms/sendTencent") @@ -79,6 +99,7 @@ public class DevSmsController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "发送小诺短信") @CommonLog("发送小诺短信") @PostMapping("/dev/sms/sendXiaonuo") @@ -93,7 +114,8 @@ public class DevSmsController { * @author xuyuxiang * @date 2022/4/24 20:00 */ - @Operation(summary = "获取短信分页") + @ApiOperationSupport(order = 5) + @Operation(summary = "获取短信分页") @GetMapping("/dev/sms/page") public CommonResult> page(DevSmsPageParam devSmsPageParam) { return CommonResult.data(devSmsService.page(devSmsPageParam)); @@ -105,11 +127,12 @@ public class DevSmsController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "删除短信") @CommonLog("删除短信") @PostMapping("/dev/sms/delete") public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List devSmsIdParamList) { + List devSmsIdParamList) { devSmsService.delete(devSmsIdParamList); return CommonResult.ok(); } @@ -120,6 +143,7 @@ public class DevSmsController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取短信详情") @GetMapping("/dev/sms/detail") public CommonResult detail(@Valid DevSmsIdParam devSmsIdParam) { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/entity/DevSms.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/entity/DevSms.java index 77d25462..62eced7d 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/entity/DevSms.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/entity/DevSms.java @@ -30,7 +30,7 @@ import vip.xiaonuo.common.pojo.CommonEntity; public class DevSms extends CommonEntity { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 短信引擎 */ diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsIdParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsIdParam.java index 90cbd521..ee05543f 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsIdParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class DevSmsIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendAliyunParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendAliyunParam.java index 8ddbdcd6..d19e08d2 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendAliyunParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendAliyunParam.java @@ -28,12 +28,12 @@ import lombok.Setter; public class DevSmsSendAliyunParam { /** 手机号 */ - @Schema(description = "手机号,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "手机号,多个逗号拼接") @NotBlank(message = "phoneNumbers不能为空") private String phoneNumbers; /** 模板编码 */ - @Schema(description = "短信服务控制台配置且审核通过的模板编码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "短信服务控制台配置且审核通过的模板编码") @NotBlank(message = "templateCode不能为空") private String templateCode; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendDynamicParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendDynamicParam.java new file mode 100644 index 00000000..36d5fab2 --- /dev/null +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendDynamicParam.java @@ -0,0 +1,43 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.dev.modular.sms.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 短信发送——动态参数 + * + * @author xuyuxiang + * @date 2022/7/31 15:21 + */ +@Getter +@Setter +public class DevSmsSendDynamicParam { + + /** 手机号 */ + @Schema(description = "手机号,多个逗号拼接") + @NotBlank(message = "phoneNumbers不能为空") + private String phoneNumbers; + + /** 模板编码或id */ + @Schema(description = "模板编码或id") + @NotBlank(message = "templateCodeOrId不能为空") + private String templateCodeOrId; + + /** 发送参数 */ + @Schema(description = "短信模板变量对应的实际值,JSON格式") + private String templateParam; +} diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendTencentParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendTencentParam.java index c7f808d7..c59a7ff7 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendTencentParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendTencentParam.java @@ -28,12 +28,12 @@ import lombok.Setter; public class DevSmsSendTencentParam { /** 手机号 */ - @Schema(description = "手机号,多个逗号拼接", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "手机号,多个逗号拼接") @NotBlank(message = "phoneNumbers不能为空") private String phoneNumbers; /** 模板编码 */ - @Schema(description = "短信服务控制台配置且审核通过的模板编码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "短信服务控制台配置且审核通过的模板编码") @NotBlank(message = "templateCode不能为空") private String templateCode; @@ -41,10 +41,6 @@ public class DevSmsSendTencentParam { @Schema(description = "短信模板变量对应的顺序。支持传入多个参数,逗号拼接") private String templateParam; - /** sdkAppId */ - @Schema(description = "在短信控制台添加应用后生成的实际SdkAppId") - private String sdkAppId; - /** 短信签名 */ @Schema(description = "短信服务控制台配置且审核通过的短信签名") private String signName; diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendXiaonuoParam.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendXiaonuoParam.java index d14d9870..a089f4b0 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendXiaonuoParam.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/param/DevSmsSendXiaonuoParam.java @@ -32,6 +32,14 @@ public class DevSmsSendXiaonuoParam { @NotBlank(message = "phoneNumbers不能为空") private String phoneNumbers; + /** 模板编码 */ + @Schema(description = "模板编码") + private String templateCode; + + /** 发送参数 */ + @Schema(description = "发送参数") + private String templateParam; + /** 短信内容 */ @Schema(description = "短信内容") private String message; @@ -39,4 +47,7 @@ public class DevSmsSendXiaonuoParam { /** 短信签名 */ @Schema(description = "短信签名") private String signName; + + + } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/provider/DevSmsApiProvider.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/provider/DevSmsApiProvider.java index c2ca7747..5835ff66 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/provider/DevSmsApiProvider.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/provider/DevSmsApiProvider.java @@ -12,10 +12,13 @@ */ package vip.xiaonuo.dev.modular.sms.provider; +import cn.hutool.json.JSONObject; +import cn.hutool.json.JSONUtil; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import vip.xiaonuo.dev.api.DevSmsApi; import vip.xiaonuo.dev.modular.sms.param.DevSmsSendAliyunParam; +import vip.xiaonuo.dev.modular.sms.param.DevSmsSendDynamicParam; import vip.xiaonuo.dev.modular.sms.param.DevSmsSendTencentParam; import vip.xiaonuo.dev.modular.sms.param.DevSmsSendXiaonuoParam; import vip.xiaonuo.dev.modular.sms.service.DevSmsService; @@ -32,6 +35,15 @@ public class DevSmsApiProvider implements DevSmsApi { @Resource private DevSmsService devSmsService; + @Override + public void sendDynamicSms(String phoneNumbers, String templateCodeOrId, JSONObject paramMap) { + DevSmsSendDynamicParam dynamicParam = new DevSmsSendDynamicParam(); + dynamicParam.setPhoneNumbers(phoneNumbers); + dynamicParam.setTemplateCodeOrId(templateCodeOrId); + dynamicParam.setTemplateParam(JSONUtil.toJsonStr(paramMap)); + devSmsService.sendDynamic(dynamicParam); + } + @Override public void sendSmsAliyun(String phoneNumbers, String signName, String templateCode, String templateParam) { DevSmsSendAliyunParam devSmsSendAliyunParam = new DevSmsSendAliyunParam(); @@ -43,9 +55,8 @@ public class DevSmsApiProvider implements DevSmsApi { } @Override - public void sendSmsTencent(String sdkAppId, String phoneNumbers, String signName, String templateCode, String templateParam) { + public void sendSmsTencent(String phoneNumbers, String signName, String templateCode, String templateParam) { DevSmsSendTencentParam devSmsSendTencentParam = new DevSmsSendTencentParam(); - devSmsSendTencentParam.setSdkAppId(sdkAppId); devSmsSendTencentParam.setPhoneNumbers(phoneNumbers); devSmsSendTencentParam.setSignName(signName); devSmsSendTencentParam.setTemplateCode(templateCode); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/DevSmsService.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/DevSmsService.java index ac334066..2f3bb311 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/DevSmsService.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/DevSmsService.java @@ -12,6 +12,7 @@ */ package vip.xiaonuo.dev.modular.sms.service; +import cn.hutool.json.JSONObject; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.IService; import vip.xiaonuo.dev.modular.sms.entity.DevSms; @@ -27,6 +28,26 @@ import java.util.List; **/ public interface DevSmsService extends IService { + /** + * 动态发送短信 + * + * @param engine 发送引擎 + * @param phoneNumbers 手机号 + * @param templateCodeOrId 模板id或编码 + * @param templateParam 发送参数 + * @author xuyuxiang + * @date 2022/2/7 22:29 + */ + void sendDynamic(String engine, String phoneNumbers, String templateCodeOrId, JSONObject templateParam); + + /** + * 动态发送短信 + * + * @author xuyuxiang + * @date 2022/6/21 18:37 + **/ + void sendDynamic(DevSmsSendDynamicParam devSmsSendDynamicParam); + /** * 发送短信——阿里云 * diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/impl/DevSmsServiceImpl.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/impl/DevSmsServiceImpl.java index 774e9735..88972c48 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/impl/DevSmsServiceImpl.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/service/impl/DevSmsServiceImpl.java @@ -14,17 +14,22 @@ package vip.xiaonuo.dev.modular.sms.service.impl; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.collection.CollStreamUtil; +import cn.hutool.core.convert.Convert; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.PhoneUtil; import cn.hutool.core.util.StrUtil; +import cn.hutool.json.JSONObject; +import cn.hutool.json.JSONUtil; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.page.CommonPageRequest; +import vip.xiaonuo.dev.api.DevConfigApi; import vip.xiaonuo.dev.modular.sms.entity.DevSms; import vip.xiaonuo.dev.modular.sms.enums.DevSmsEngineTypeEnum; import vip.xiaonuo.dev.modular.sms.mapper.DevSmsMapper; @@ -34,6 +39,7 @@ import vip.xiaonuo.dev.modular.sms.util.DevSmsAliyunUtil; import vip.xiaonuo.dev.modular.sms.util.DevSmsTencentUtil; import vip.xiaonuo.dev.modular.sms.util.DevSmsXiaonuoUtil; +import java.util.LinkedHashMap; import java.util.List; /** @@ -45,6 +51,52 @@ import java.util.List; @Service public class DevSmsServiceImpl extends ServiceImpl implements DevSmsService { + /** 默认短信引擎 */ + private static final String SNOWY_SYS_DEFAULT_SMS_ENGINE_KEY = "SNOWY_SYS_DEFAULT_SMS_ENGINE"; + + @Resource + private DevConfigApi devConfigApi; + + @Transactional(rollbackFor = Exception.class) + @Override + public void sendDynamic(String engine, String phoneNumbers, String templateCodeOrId, JSONObject templateParam) { + if(engine.equals(DevSmsEngineTypeEnum.XIAONUO.getValue())) { + DevSmsSendXiaonuoParam devSmsSendXiaonuoParam = new DevSmsSendXiaonuoParam(); + devSmsSendXiaonuoParam.setPhoneNumbers(phoneNumbers); + devSmsSendXiaonuoParam.setTemplateCode(templateCodeOrId); + devSmsSendXiaonuoParam.setTemplateParam(JSONUtil.toJsonStr(templateParam)); + this.sendXiaonuo(devSmsSendXiaonuoParam); + } else if (engine.equals(DevSmsEngineTypeEnum.ALIYUN.getValue())) { + DevSmsSendAliyunParam devSmsSendAliyunParam = new DevSmsSendAliyunParam(); + devSmsSendAliyunParam.setPhoneNumbers(phoneNumbers); + devSmsSendAliyunParam.setTemplateCode(templateCodeOrId); + devSmsSendAliyunParam.setTemplateParam(JSONUtil.toJsonStr(templateParam)); + this.sendAliyun(devSmsSendAliyunParam); + } else if (engine.equals(DevSmsEngineTypeEnum.TENCENT.getValue())) { + DevSmsSendTencentParam devSmsSendTencentParam = new DevSmsSendTencentParam(); + devSmsSendTencentParam.setPhoneNumbers(phoneNumbers); + devSmsSendTencentParam.setTemplateCode(templateCodeOrId); + devSmsSendTencentParam.setTemplateParam(JSONUtil.toJsonStr(templateParam)); + this.sendTencent(devSmsSendTencentParam); + } else { + throw new CommonException("不支持的短信引擎:{}", engine); + } + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void sendDynamic(DevSmsSendDynamicParam devSmsSendDynamicParam) { + String defaultSmsEngine = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_SMS_ENGINE_KEY); + if(ObjectUtil.isEmpty(defaultSmsEngine)) { + throw new CommonException("请联系管理员配置默认短信发送引擎"); + } + String phoneNumbers = devSmsSendDynamicParam.getPhoneNumbers(); + String templateCodeOrId = devSmsSendDynamicParam.getTemplateCodeOrId(); + String templateParamStr = devSmsSendDynamicParam.getTemplateParam(); + JSONObject templateParam = JSONUtil.parseObj(templateParamStr); + this.sendDynamic(defaultSmsEngine, phoneNumbers, templateCodeOrId, templateParam); + } + @Transactional(rollbackFor = Exception.class) @Override public void sendAliyun(DevSmsSendAliyunParam devSmsSendAliyunParam) { @@ -63,7 +115,7 @@ public class DevSmsServiceImpl extends ServiceImpl impleme @Override public void sendTencent(DevSmsSendTencentParam smsSendTencentParam) { validPhone(smsSendTencentParam.getPhoneNumbers()); - String receiptInfo = DevSmsTencentUtil.sendSms(smsSendTencentParam.getSdkAppId(), smsSendTencentParam.getPhoneNumbers(), + String receiptInfo =DevSmsTencentUtil.sendSms(smsSendTencentParam.getPhoneNumbers(), smsSendTencentParam.getSignName(), smsSendTencentParam.getTemplateCode(), smsSendTencentParam.getTemplateParam()); DevSms devSms = new DevSms(); BeanUtil.copyProperties(smsSendTencentParam, devSms); @@ -76,8 +128,16 @@ public class DevSmsServiceImpl extends ServiceImpl impleme @Override public void sendXiaonuo(DevSmsSendXiaonuoParam devSmsSendXiaonuoParam) { validPhone(devSmsSendXiaonuoParam.getPhoneNumbers()); - String receiptInfo = DevSmsXiaonuoUtil.sendSms(devSmsSendXiaonuoParam.getPhoneNumbers(), devSmsSendXiaonuoParam.getSignName(), - devSmsSendXiaonuoParam.getMessage()); + String receiptInfo; + if(ObjectUtil.isEmpty(devSmsSendXiaonuoParam.getTemplateCode())) { + receiptInfo = DevSmsXiaonuoUtil.sendSms(devSmsSendXiaonuoParam.getPhoneNumbers(), devSmsSendXiaonuoParam.getSignName(), + devSmsSendXiaonuoParam.getMessage()); + } else { + LinkedHashMap paramMap = new LinkedHashMap<>(); + JSONUtil.parseObj(devSmsSendXiaonuoParam.getTemplateParam()).forEach((k, v) -> paramMap.put(k, Convert.toStr(v))); + receiptInfo = DevSmsXiaonuoUtil.sendSms(devSmsSendXiaonuoParam.getPhoneNumbers(), devSmsSendXiaonuoParam.getSignName(), + devSmsSendXiaonuoParam.getTemplateCode(), paramMap); + } DevSms devSms = new DevSms(); BeanUtil.copyProperties(devSmsSendXiaonuoParam, devSms); devSms.setSignName(ObjectUtil.isNotEmpty(devSms.getSignName())?devSms.getSignName():DevSmsXiaonuoUtil.getDefaultSignName()); @@ -103,6 +163,7 @@ public class DevSmsServiceImpl extends ServiceImpl impleme return this.page(CommonPageRequest.defaultPage(), queryWrapper); } + @Transactional(rollbackFor = Exception.class) @Override public void delete(List devSmsIdParamList) { this.removeByIds(CollStreamUtil.toList(devSmsIdParamList, DevSmsIdParam::getId)); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsAliyunUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsAliyunUtil.java index 1a302aad..303ccad6 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsAliyunUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsAliyunUtil.java @@ -12,21 +12,27 @@ */ package vip.xiaonuo.dev.modular.sms.util; +import cn.hutool.core.collection.CollectionUtil; +import cn.hutool.core.convert.Convert; import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; -import com.aliyun.dysmsapi20170525.Client; -import com.aliyun.dysmsapi20170525.models.SendSmsRequest; -import com.aliyun.dysmsapi20170525.models.SendSmsResponse; -import com.aliyun.dysmsapi20170525.models.SendSmsResponseBody; -import com.aliyun.teaopenapi.models.Config; import lombok.extern.slf4j.Slf4j; +import org.dromara.sms4j.aliyun.config.AlibabaConfig; +import org.dromara.sms4j.api.SmsBlend; +import org.dromara.sms4j.api.entity.SmsResponse; +import org.dromara.sms4j.core.factory.SmsFactory; +import org.dromara.sms4j.javase.config.SEInitializer; +import org.dromara.sms4j.provider.config.SmsConfig; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.dev.api.DevConfigApi; +import java.util.LinkedHashMap; + /** * 阿里云短信工具类 - * 参考文档:https://next.api.aliyun.com/api-tools/sdk/Dysmsapi?version=2017-05-25&language=java-tea * * @author xuyuxiang * @date 2022/1/2 17:05 @@ -34,11 +40,10 @@ import vip.xiaonuo.dev.api.DevConfigApi; @Slf4j public class DevSmsAliyunUtil { - private static Client client; + private static SmsBlend smsBlend; private static final String SNOWY_SMS_ALIYUN_ACCESS_KEY_ID_KEY = "SNOWY_SMS_ALIYUN_ACCESS_KEY_ID"; private static final String SNOWY_SMS_ALIYUN_ACCESS_KEY_SECRET_KEY = "SNOWY_SMS_ALIYUN_ACCESS_KEY_SECRET"; - private static final String SNOWY_SMS_ALIYUN_END_POINT_KEY = "SNOWY_SMS_ALIYUN_END_POINT"; private static final String SNOWY_SMS_ALIYUN_DEFAULT_SIGN_NAME_KEY = "SNOWY_SMS_ALIYUN_DEFAULT_SIGN_NAME"; /** @@ -47,7 +52,7 @@ public class DevSmsAliyunUtil { * @author xuyuxiang * @date 2022/1/5 23:24 */ - private static void initClient() { + private static void initClient(String signName) { DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); @@ -65,50 +70,62 @@ public class DevSmsAliyunUtil { throw new CommonException("阿里云短信操作客户端未正确配置:accessKeySecret为空"); } - /* endpoint */ - String endpoint = devConfigApi.getValueByKey(SNOWY_SMS_ALIYUN_END_POINT_KEY); - - if(ObjectUtil.isEmpty(endpoint)) { - throw new CommonException("阿里云短信操作客户端未正确配置:endpoint为空"); - } - - try { - client = new Client(new Config().setAccessKeyId(accessKeyId).setAccessKeySecret(accessKeySecret).setEndpoint(endpoint)); - } catch (Exception e) { - throw new CommonException(e.getMessage()); - } + AlibabaConfig alibabaConfig = new AlibabaConfig(); + alibabaConfig.setConfigId(accessKeyId); + alibabaConfig.setAccessKeyId(accessKeyId); + alibabaConfig.setAccessKeySecret(accessKeySecret); + alibabaConfig.setSignature(signName); + SEInitializer.initializer().fromConfig(new SmsConfig(), CollectionUtil.newArrayList(alibabaConfig)); + smsBlend = SmsFactory.getSmsBlend(alibabaConfig.getConfigId()); } /** - * 发送短信 + * 发送模板短信 * * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 - * 上限为1000个手机号码。批量调用相对于单条调用及时性稍有延迟。 - * @param signName 短信服务控制台配置且审核通过的短信签名,为空则使用默认签名 - * @param templateCode 短信服务控制台配置且审核通过的模板编码 + * @param signName 短信签名,为空则使用默认签名 + * @param templateId 模板id * @param templateParam 短信模板变量对应的实际值,JSON格式。支持传入多个参数,示例:{"name":"张三","number":"15038****76"} - * @return 发送的结果信息集合 com.aliyun.dysmsapi20170525.models.SendSmsResponse + * @return 发送的结果信息 * @author xuyuxiang * @date 2022/2/24 13:42 **/ - public static String sendSms(String phoneNumbers, String signName, String templateCode, String templateParam) { + public static String sendSms(String phoneNumbers, String signName, String templateId, String templateParam) { + LinkedHashMap paramMap = new LinkedHashMap<>(); + JSONUtil.parseObj(templateParam).forEach((k, v) -> paramMap.put(k, Convert.toStr(v))); + return sendSms(phoneNumbers, signName, templateId, paramMap); + } + + /** + * 发送模板短信 + * + * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 + * @param signName 短信签名,为空则使用默认签名 + * @param templateId 模板id + * @param paramMap 短信参数,HashMap + * @return 发送的结果信息 + * @author xuyuxiang + * @date 2022/2/24 13:42 + **/ + public static String sendSms(String phoneNumbers, String signName, String templateId, LinkedHashMap paramMap) { try { - initClient(); if(ObjectUtil.isEmpty(signName)) { signName = getDefaultSignName(); } - SendSmsRequest sendSmsRequest = new SendSmsRequest() - .setPhoneNumbers(phoneNumbers) - .setSignName(signName) - .setTemplateCode(templateCode) - .setTemplateParam(templateParam); - SendSmsResponse sendSmsResponse = client.sendSms(sendSmsRequest); - SendSmsResponseBody body = sendSmsResponse.getBody(); - String code = body.getCode().toLowerCase(); - if("ok".equals(code)) { - return JSONUtil.toJsonStr(body); + // 初始化客户端 + initClient(signName); + // 发送短信 + SmsResponse smsResponse = smsBlend.massTexting(StrUtil.split(phoneNumbers, StrUtil.COMMA), templateId, paramMap); + if(smsResponse.isSuccess()) { + return JSONUtil.toJsonStr(smsResponse.getData()); } else { - throw new CommonException(body.getMessage()); + String data = Convert.toStr(smsResponse.getData()); + if(JSONUtil.isTypeJSON(data)) { + JSONObject responseData = JSONUtil.parseObj(smsResponse.getData()); + throw new CommonException(responseData.getStr("resInfo")); + } else { + throw new CommonException(data); + } } } catch (Exception e) { throw new CommonException(e.getMessage()); diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsTencentUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsTencentUtil.java index b400f072..d5fce517 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsTencentUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsTencentUtil.java @@ -12,22 +12,28 @@ */ package vip.xiaonuo.dev.modular.sms.util; +import cn.hutool.core.collection.CollectionUtil; +import cn.hutool.core.convert.Convert; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; -import com.tencentcloudapi.common.Credential; -import com.tencentcloudapi.sms.v20210111.SmsClient; -import com.tencentcloudapi.sms.v20210111.models.SendSmsRequest; -import com.tencentcloudapi.sms.v20210111.models.SendSmsResponse; -import com.tencentcloudapi.sms.v20210111.models.SendStatus; import lombok.extern.slf4j.Slf4j; +import org.dromara.sms4j.api.SmsBlend; +import org.dromara.sms4j.api.entity.SmsResponse; +import org.dromara.sms4j.core.factory.SmsFactory; +import org.dromara.sms4j.javase.config.SEInitializer; +import org.dromara.sms4j.provider.config.SmsConfig; +import org.dromara.sms4j.tencent.config.TencentConfig; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.dev.api.DevConfigApi; +import java.util.LinkedHashMap; +import java.util.List; + /** * 腾讯云短信工具类 - * 参考文档:https://cloud.tencent.com/document/product/382/43194 * * @author xuyuxiang * @date 2022/1/2 17:05 @@ -35,11 +41,10 @@ import vip.xiaonuo.dev.api.DevConfigApi; @Slf4j public class DevSmsTencentUtil { - private static SmsClient client; + private static SmsBlend smsBlend; private static final String SNOWY_SMS_TENCENT_SECRET_ID_KEY = "SNOWY_SMS_TENCENT_SECRET_ID"; private static final String SNOWY_SMS_TENCENT_SECRET_KEY_KEY = "SNOWY_SMS_TENCENT_SECRET_KEY"; - private static final String SNOWY_SMS_TENCENT_REGION_ID_KEY = "SNOWY_SMS_TENCENT_REGION_ID"; private static final String SNOWY_SMS_TENCENT_DEFAULT_SDK_APP_ID_KEY = "SNOWY_SMS_TENCENT_DEFAULT_SDK_APP_ID"; private static final String SNOWY_SMS_TENCENT_DEFAULT_SIGN_NAME_KEY = "SNOWY_SMS_TENCENT_DEFAULT_SIGN_NAME"; @@ -49,7 +54,7 @@ public class DevSmsTencentUtil { * @author xuyuxiang * @date 2022/1/5 23:24 */ - private static void initClient() { + private static void initClient(String signName) { DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); @@ -67,52 +72,76 @@ public class DevSmsTencentUtil { throw new CommonException("腾讯云短信操作客户端未正确配置:secretKey为空"); } - /* regionId */ - String regionId = devConfigApi.getValueByKey(SNOWY_SMS_TENCENT_REGION_ID_KEY); - - if(ObjectUtil.isEmpty(regionId)) { - throw new CommonException("腾讯云短信操作客户端未正确配置:regionId为空"); + /* sdkAppId */ + String sdkAppId = devConfigApi.getValueByKey(SNOWY_SMS_TENCENT_DEFAULT_SDK_APP_ID_KEY); + if(ObjectUtil.isEmpty(sdkAppId)) { + throw new CommonException("腾讯云短信操作客户端未正确配置:sdkAppId为空"); } - - client = new SmsClient(new Credential(secretId, secretKey), regionId); + TencentConfig tencentConfig = new TencentConfig(); + tencentConfig.setConfigId(secretId); + tencentConfig.setAccessKeyId(secretId); + tencentConfig.setAccessKeySecret(secretKey); + tencentConfig.setSignature(signName); + tencentConfig.setSdkAppId(sdkAppId); + SEInitializer.initializer().fromConfig(new SmsConfig(), CollectionUtil.newArrayList(tencentConfig)); + smsBlend = SmsFactory.getSmsBlend(tencentConfig.getConfigId()); } /** - * 发送短信 + * 发送模板短信 * - * @param sdkAppId 短信 SdkAppId,在 短信控制台 添加应用后生成的实际 SdkAppId,示例如1400006666。 - * 可前往 [短信控制台](https://console.cloud.tencent.com/smsv2/app-manage) 查看 * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 - * 上限为1000个手机号码。批量调用相对于单条调用及时性稍有延迟。 - * @param signName 短信服务控制台配置且审核通过的短信签名,为空则使用默认签名 - * @param templateCode 短信服务控制台配置且审核通过的模板编码 - * @param templateParam 短信模板变量对应的顺序。支持传入多个参数,逗号拼接,示例:"张三,15038****76,进行中" - * @return 发送的结果信息集合 com.tencentcloudapi.sms.v20210111.models.SendStatus + * @param signName 短信签名,为空则使用默认签名 + * @param templateId 模板id + * @param templateParam 短信模板变量对应的顺序。支持传入多个参数,逗号拼接,示例:"张三,15038****76,进行中", + * 同时支持JSON格式,示例:{"name":"张三","number":"15038****76"},此时参数的key无效,仅方便统一格式 + * @return 发送的结果信息 * @author xuyuxiang * @date 2022/2/24 13:42 **/ - public static String sendSms(String sdkAppId, String phoneNumbers, String signName, String templateCode, String templateParam) { - try { - initClient(); - if(ObjectUtil.isEmpty(sdkAppId)) { - sdkAppId = getDefaultSdkAppId(); + public static String sendSms(String phoneNumbers, String signName, String templateId, String templateParam) { + LinkedHashMap paramMap = new LinkedHashMap<>(); + if(JSONUtil.isTypeJSON(templateParam)) { + JSONUtil.parseObj(templateParam).forEach((k, v) -> paramMap.put(k, Convert.toStr(v))); + } else { + List templateParamList = StrUtil.split(templateParam, StrUtil.COMMA); + for (int i = 0; i < templateParamList.size(); i++) { + paramMap.put(Convert.toStr(i + 1), templateParamList.get(i)); } + } + return sendSms(phoneNumbers, signName, templateId, paramMap); + } + + /** + * 发送模板短信 + * + * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 + * @param signName 短信签名,为空则使用默认签名 + * @param templateId 模板id + * @param paramMap 短信参数,HashMap + * @return 发送的结果信息 + * @author xuyuxiang + * @date 2022/2/24 13:42 + **/ + public static String sendSms(String phoneNumbers, String signName, String templateId, LinkedHashMap paramMap) { + try { if(ObjectUtil.isEmpty(signName)) { signName = getDefaultSignName(); } - SendSmsRequest sendSmsRequest = new SendSmsRequest(); - sendSmsRequest.setSmsSdkAppId(sdkAppId); - sendSmsRequest.setPhoneNumberSet(StrUtil.splitToArray(phoneNumbers, StrUtil.COMMA)); - sendSmsRequest.setSignName(signName); - sendSmsRequest.setTemplateId(templateCode); - sendSmsRequest.setTemplateParamSet(ObjectUtil.isNotEmpty(templateParam)?StrUtil.splitToArray(templateParam, StrUtil.COMMA):null); - SendSmsResponse sendSmsResponse = client.SendSms(sendSmsRequest); - SendStatus sendStatus = sendSmsResponse.getSendStatusSet()[0]; - String code = sendStatus.getCode().toLowerCase(); - if("ok".equals(code)) { - return JSONUtil.toJsonStr(sendSmsResponse); + // 初始化客户端 + initClient(signName); + // 发送短信 + SmsResponse smsResponse = smsBlend.massTexting(StrUtil.split(phoneNumbers, StrUtil.COMMA), templateId, paramMap); + if(smsResponse.isSuccess()) { + return JSONUtil.toJsonStr(smsResponse.getData()); } else { - throw new CommonException(sendStatus.getMessage()); + String data = Convert.toStr(smsResponse.getData()); + if(JSONUtil.isTypeJSON(data)) { + JSONObject responseData = JSONUtil.parseObj(smsResponse.getData()); + throw new CommonException(responseData.getStr("resInfo")); + } else { + throw new CommonException(data); + } } } catch (Exception e) { throw new CommonException(e.getMessage()); @@ -134,20 +163,4 @@ public class DevSmsTencentUtil { } return signName; } - - /** - * 获取默认sdkAppId - * - * @author xuyuxiang - * @date 2024/1/26 16:40 - **/ - public static String getDefaultSdkAppId() { - // sdkAppId为空,则获取默认sdkAppId - DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); - String sdkAppId = devConfigApi.getValueByKey(SNOWY_SMS_TENCENT_DEFAULT_SDK_APP_ID_KEY); - if(ObjectUtil.isEmpty(sdkAppId)) { - throw new CommonException("腾讯云短信操作客户端未正确配置:sdkAppId为空"); - } - return sdkAppId; - } } diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsXiaonuoUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsXiaonuoUtil.java index cf238652..c99f859e 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsXiaonuoUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sms/util/DevSmsXiaonuoUtil.java @@ -13,6 +13,7 @@ package vip.xiaonuo.dev.modular.sms.util; import cn.hutool.core.collection.CollectionUtil; +import cn.hutool.core.convert.Convert; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.extra.spring.SpringUtil; @@ -28,6 +29,8 @@ import org.dromara.sms4j.provider.config.SmsConfig; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.dev.api.DevConfigApi; +import java.util.LinkedHashMap; + /** * 小诺短信工具类 * @@ -76,21 +79,21 @@ public class DevSmsXiaonuoUtil { } DingZhongConfig dingZhongConfig = new DingZhongConfig(); + dingZhongConfig.setConfigId(accessKeyId); dingZhongConfig.setAccessKeyId(accessKeyId); dingZhongConfig.setAccessKeySecret(accessKeySecret); - dingZhongConfig.setRequestUrl(requestUrl); dingZhongConfig.setSignature(signName); - dingZhongConfig.setConfigId("XIAONUO"); + dingZhongConfig.setRequestUrl(requestUrl); SEInitializer.initializer().fromConfig(new SmsConfig(), CollectionUtil.newArrayList(dingZhongConfig)); smsBlend = SmsFactory.getSmsBlend(dingZhongConfig.getConfigId()); } /** - * 发送短信 + * 发送纯文字短信 * * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 * @param signName 短信签名,为空则使用默认签名 - * @param message 短信内容 + * @param message 短信内容,自定义文字 * @return 发送的结果信息 * @author xuyuxiang * @date 2022/2/24 13:42 @@ -104,6 +107,42 @@ public class DevSmsXiaonuoUtil { initClient(signName); // 发送短信 SmsResponse smsResponse = smsBlend.massTexting(StrUtil.split(phoneNumbers, StrUtil.COMMA), message); + if(smsResponse.isSuccess()) { + return JSONUtil.toJsonStr(smsResponse.getData()); + } else { + String data = Convert.toStr(smsResponse.getData()); + if(JSONUtil.isTypeJSON(data)) { + JSONObject responseData = JSONUtil.parseObj(smsResponse.getData()); + throw new CommonException(responseData.getStr("resInfo")); + } else { + throw new CommonException(data); + } + } + } catch (Exception e) { + throw new CommonException(e.getMessage()); + } + } + + /** + * 发送模板短信 + * + * @param phoneNumbers 手机号码,支持对多个手机号码发送短信,手机号码之间以半角逗号(,)分隔。 + * @param signName 短信签名,为空则使用默认签名 + * @param templateId 模板id + * @param paramMap 短信参数 + * @return 发送的结果信息 + * @author xuyuxiang + * @date 2022/2/24 13:42 + **/ + public static String sendSms(String phoneNumbers, String signName, String templateId, LinkedHashMap paramMap) { + try { + if(ObjectUtil.isEmpty(signName)) { + signName = getDefaultSignName(); + } + // 初始化客户端 + initClient(signName); + // 发送短信 + SmsResponse smsResponse = smsBlend.massTexting(StrUtil.split(phoneNumbers, StrUtil.COMMA), templateId, paramMap); if(smsResponse.isSuccess()) { return JSONUtil.toJsonStr(smsResponse.getData()); } else { diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/controller/DevSseEmitterController.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/controller/DevSseEmitterController.java index 84126148..9bb89bc1 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/controller/DevSseEmitterController.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/controller/DevSseEmitterController.java @@ -12,6 +12,8 @@ */ package vip.xiaonuo.dev.modular.sse.controller; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; @@ -30,6 +32,7 @@ import java.util.function.Consumer; * @date 2023/7/3 **/ @Tag(name = "SSE通信控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 10) @RestController @Validated public class DevSseEmitterController { @@ -43,6 +46,7 @@ public class DevSseEmitterController { * @author diantu * @date 2023/7/3 **/ + @ApiOperationSupport(order = 1) @Operation(summary = "创建sse连接") @GetMapping("/dev/sse/createConnect") public SseEmitter createConnect(String clientId, @@ -58,6 +62,7 @@ public class DevSseEmitterController { * @author diantu * @date 2023/7/3 **/ + @ApiOperationSupport(order = 2) @Operation(summary = "关闭sse连接") @GetMapping("/dev/sse/closeSseConnect") public void closeSseConnect(String clientId){ @@ -70,6 +75,7 @@ public class DevSseEmitterController { * @author diantu * @date 2023/7/3 **/ + @ApiOperationSupport(order = 3) @Operation(summary = "推送消息到所有客户端") @PostMapping("/dev/sse/broadcast") public void sendMessageToAllClient(@RequestBody(required = false) String msg){ @@ -82,6 +88,7 @@ public class DevSseEmitterController { * @author diantu * @date 2023/7/3 **/ + @ApiOperationSupport(order = 4) @Operation(summary = "根据clientId发送消息给某一客户端") @PostMapping("/dev/sse/sendMessage") public void sendMessageToOneClient(String clientId,String msg){ diff --git a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/util/DevSseCacheUtil.java b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/util/DevSseCacheUtil.java index 6ef15321..854778b7 100644 --- a/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/util/DevSseCacheUtil.java +++ b/snowy-plugin/snowy-plugin-dev/src/main/java/vip/xiaonuo/dev/modular/sse/util/DevSseCacheUtil.java @@ -38,7 +38,7 @@ public class DevSseCacheUtil { /** * 创建一个容器来存储所有的 SseEmitter(使用ConcurrentHashMap是因为它是线程安全的)。 */ - public static Map> sseCache = new ConcurrentHashMap<>(); + public static Map> sseCache = new ConcurrentHashMap<>(); /** @@ -48,7 +48,7 @@ public class DevSseCacheUtil { * @date 2023/7/3 **/ public static SseEmitter getSseEmitterByClientId(String clientId) { - Map map = sseCache.get(clientId); + Map map = sseCache.get(clientId); if (map == null || map.isEmpty()) { return null; } @@ -62,7 +62,7 @@ public class DevSseCacheUtil { * @date 2023/7/18 **/ public static ScheduledFuture getSseFutureByClientId(String clientId) { - Map map = sseCache.get(clientId); + Map map = sseCache.get(clientId); if (map == null || map.isEmpty()) { return null; } @@ -76,7 +76,7 @@ public class DevSseCacheUtil { * @date 2023/7/18 **/ public static ScheduledFuture getLoginIdByClientId(String clientId) { - Map map = sseCache.get(clientId); + Map map = sseCache.get(clientId); if (map == null || map.isEmpty()) { return null; } @@ -89,12 +89,12 @@ public class DevSseCacheUtil { * @author diantu * @date 2023/7/18 **/ - public static String getClientIdByLoginId(String loginId){ - if(existSseCache()){ + public static String getClientIdByLoginId(String loginId) { + if (existSseCache()) { for (Map.Entry> entry : sseCache.entrySet()) { - Map map = sseCache.get(entry.getKey()); + Map map = sseCache.get(entry.getKey()); String lId = (String) map.get(DevSseEmitterParameterEnum.LOGINID.getValue()); - if(loginId.equals(lId)){ + if (loginId.equals(lId)) { return entry.getKey(); } } @@ -109,7 +109,7 @@ public class DevSseCacheUtil { * @date 2023/7/3 **/ public static boolean existSseCache() { - return sseCache.size()>0; + return !sseCache.isEmpty(); } /** @@ -118,8 +118,8 @@ public class DevSseCacheUtil { * @author diantu * @date 2023/7/3 **/ - public static boolean connectionValidity(String clientId,String loginId){ - if(sseCache.get(clientId) == null){ + public static boolean connectionValidity(String clientId, String loginId) { + if (sseCache.get(clientId) == null) { return false; } return Objects.equals(loginId, sseCache.get(clientId).get(DevSseEmitterParameterEnum.LOGINID.getValue())); @@ -131,14 +131,14 @@ public class DevSseCacheUtil { * @author diantu * @date 2023/7/3 **/ - public static void addConnection(String clientId,String loginId, SseEmitter emitter, ScheduledFuture future) { + public static void addConnection(String clientId, String loginId, SseEmitter emitter, ScheduledFuture future) { final SseEmitter oldEmitter = getSseEmitterByClientId(clientId); if (oldEmitter != null) { - throw new CommonException("连接已存在:{}",clientId); + throw new CommonException("连接已存在:{}", clientId); } - Map map = new ConcurrentHashMap<>(); - map.put(DevSseEmitterParameterEnum.EMITTER.getValue(),emitter); - if(future!=null){ + Map map = new ConcurrentHashMap<>(); + map.put(DevSseEmitterParameterEnum.EMITTER.getValue(), emitter); + if (future != null) { map.put(DevSseEmitterParameterEnum.FUTURE.getValue(), future); } map.put(DevSseEmitterParameterEnum.LOGINID.getValue(), loginId); @@ -166,7 +166,7 @@ public class DevSseCacheUtil { * @author diantu * @date 2023/7/3 */ - public static void cancelScheduledFuture(String clientId){ + public static void cancelScheduledFuture(String clientId) { ScheduledFuture future = getSseFutureByClientId(clientId); if (future != null) { future.cancel(true); @@ -194,8 +194,8 @@ public class DevSseCacheUtil { * @author diantu * @date 2023/7/3 **/ - public static Runnable timeoutCallBack(String clientId){ - return ()->{ + public static Runnable timeoutCallBack(String clientId) { + return () -> { log.info("连接超时:{}", clientId); removeConnection(clientId); cancelScheduledFuture(clientId); @@ -227,11 +227,11 @@ public class DevSseCacheUtil { return; } // 判断发送的消息是否为空 - if (StrUtil.isEmpty(msg)){ + if (StrUtil.isEmpty(msg)) { log.info("群发消息为空"); return; } - CommonResult message = new CommonResult<>(CommonResult.CODE_SUCCESS,"",msg); + CommonResult message = new CommonResult<>(CommonResult.CODE_SUCCESS, "", msg); for (Map.Entry> entry : sseCache.entrySet()) { sendMessageToClientByClientId(entry.getKey(), message); } @@ -244,16 +244,16 @@ public class DevSseCacheUtil { * @date 2023/7/3 **/ public static void sendMessageToOneClient(String clientId, String msg) { - if (StrUtil.isEmpty(clientId)){ + if (StrUtil.isEmpty(clientId)) { log.info("客户端ID为空"); return; } - if (StrUtil.isEmpty(msg)){ - log.info("向客户端{}推送消息为空",clientId); + if (StrUtil.isEmpty(msg)) { + log.info("向客户端{}推送消息为空", clientId); return; } - CommonResult message = new CommonResult<>(CommonResult.CODE_SUCCESS,"",msg); - sendMessageToClientByClientId(clientId,message); + CommonResult message = new CommonResult<>(CommonResult.CODE_SUCCESS, "", msg); + sendMessageToClientByClientId(clientId, message); } /** @@ -264,16 +264,16 @@ public class DevSseCacheUtil { **/ public static void sendMessageToClientByClientId(String clientId, CommonResult message) { Map map = sseCache.get(clientId); - if (map==null||map.size()==0) { - log.error("推送消息失败:客户端{}未创建长链接,失败消息:{}",clientId, message.toString()); + if (map == null || map.isEmpty()) { + log.error("推送消息失败:客户端{}未创建长链接,失败消息:{}", clientId, message.toString()); return; } - SseEmitter.SseEventBuilder sendData = SseEmitter.event().data(message,MediaType.APPLICATION_JSON); + SseEmitter.SseEventBuilder sendData = SseEmitter.event().data(message, MediaType.APPLICATION_JSON); SseEmitter sseEmitter = getSseEmitterByClientId(clientId); try { Objects.requireNonNull(sseEmitter).send(sendData); } catch (Exception e) { - log.error("推送消息失败,报错异常:",e); + log.error("推送消息失败,报错异常:", e); removeConnection(clientId); } } diff --git a/snowy-plugin/snowy-plugin-gen/src/main/java/vip/xiaonuo/gen/modular/basic/entity/GenBasic.java b/snowy-plugin/snowy-plugin-gen/src/main/java/vip/xiaonuo/gen/modular/basic/entity/GenBasic.java index e9cf89be..3094270f 100644 --- a/snowy-plugin/snowy-plugin-gen/src/main/java/vip/xiaonuo/gen/modular/basic/entity/GenBasic.java +++ b/snowy-plugin/snowy-plugin-gen/src/main/java/vip/xiaonuo/gen/modular/basic/entity/GenBasic.java @@ -69,7 +69,7 @@ public class GenBasic extends CommonEntity { /** 移动端所属模块 */ @Schema(description = "移动端所属模块") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String mobileModule; /** 功能名 */ diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/mobile/MobileApiProvider.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/mobile/MobileApiProvider.java new file mode 100644 index 00000000..bf6ee047 --- /dev/null +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/mobile/MobileApiProvider.java @@ -0,0 +1,27 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.mobile.modular.mobile; + +import org.springframework.stereotype.Service; +import vip.xiaonuo.mobile.api.MobileApi; + +/** + * 移动端综合API接口实现类 + * + * @author xuyuxiang + * @date 2022/9/26 14:24 + **/ +@Service +public class MobileApiProvider implements MobileApi { + +} diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileButtonController.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileButtonController.java index e2b24f77..4c1710a1 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileButtonController.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileButtonController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.mobile.modular.resource.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -32,6 +33,7 @@ import vip.xiaonuo.mobile.modular.resource.param.button.MobileButtonIdParam; import vip.xiaonuo.mobile.modular.resource.param.button.MobileButtonPageParam; import vip.xiaonuo.mobile.modular.resource.service.MobileButtonService; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +43,7 @@ import java.util.List; * @date 2022/6/27 13:56 **/ @Tag(name = "移动端按钮控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 3) @RestController @Validated public class MobileButtonController { @@ -54,6 +57,7 @@ public class MobileButtonController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取移动端按钮分页") @GetMapping("/mobile/button/page") public CommonResult> page(MobileButtonPageParam mobileButtonPageParam) { @@ -66,6 +70,7 @@ public class MobileButtonController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加移动端按钮") @CommonLog("添加移动端按钮") @PostMapping("/mobile/button/add") @@ -80,6 +85,7 @@ public class MobileButtonController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑移动端按钮") @CommonLog("编辑移动端按钮") @PostMapping("/mobile/button/edit") @@ -94,6 +100,7 @@ public class MobileButtonController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除移动端按钮") @CommonLog("删除移动端按钮") @PostMapping("/mobile/button/delete") @@ -108,6 +115,7 @@ public class MobileButtonController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取移动端按钮详情") @GetMapping("/mobile/button/detail") public CommonResult detail(@Valid MobileButtonIdParam mobileButtonIdParam) { diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileMenuController.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileMenuController.java index 93c8d694..d90f6b6b 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileMenuController.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileMenuController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.mobile.modular.resource.controller; import cn.hutool.core.lang.tree.Tree; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -30,6 +31,7 @@ import vip.xiaonuo.mobile.modular.resource.entity.MobileModule; import vip.xiaonuo.mobile.modular.resource.param.menu.*; import vip.xiaonuo.mobile.modular.resource.service.MobileMenuService; +import javax.validation.Valid; import java.util.List; /** @@ -39,6 +41,7 @@ import java.util.List; * @date 2023/01/28 22:42 */ @Tag(name = "移动端菜单控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 1) @RestController @Validated public class MobileMenuController { @@ -52,6 +55,7 @@ public class MobileMenuController { * @author yubaoshan * @date 2023/01/28 22:42 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取移动端菜单tree") @GetMapping("/mobile/menu/tree") public CommonResult>> tree(MobileMenuTreeParam mobileMenuTreeParam) { @@ -64,6 +68,7 @@ public class MobileMenuController { * @author yubaoshan * @date 2023/01/28 22:42 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加移动端菜单") @CommonLog("添加移动端菜单") @PostMapping("/mobile/menu/add") @@ -78,6 +83,7 @@ public class MobileMenuController { * @author yubaoshan * @date 2023/01/28 22:42 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑移动端菜单") @CommonLog("编辑移动端菜单") @PostMapping("/mobile/menu/edit") @@ -92,6 +98,7 @@ public class MobileMenuController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 5) @Operation(summary = "更改移动端菜单所属模块") @CommonLog("更改移动端菜单所属模块") @PostMapping("/mobile/menu/changeModule") @@ -106,6 +113,7 @@ public class MobileMenuController { * @author yubaoshan * @date 2023/01/28 22:42 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除移动端菜单") @CommonLog("删除移动端菜单") @PostMapping("/mobile/menu/delete") @@ -121,6 +129,7 @@ public class MobileMenuController { * @author yubaoshan * @date 2023/01/28 22:42 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取移动端菜单详情") @GetMapping("/mobile/menu/detail") public CommonResult detail(@Valid MobileMenuIdParam mobileMenuIdParam) { @@ -135,6 +144,7 @@ public class MobileMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取模块选择器") @GetMapping("/mobile/menu/moduleSelector") public CommonResult> moduleSelector(MobileMenuSelectorModuleParam mobileMenuSelectorModuleParam) { @@ -147,6 +157,7 @@ public class MobileMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取菜单树选择器") @GetMapping("/mobile/menu/menuTreeSelector") public CommonResult>> menuTreeSelector(MobileMenuSelectorMenuParam mobileMenuSelectorMenuParam) { diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileModuleController.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileModuleController.java index a08da7d8..9a41f8f4 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileModuleController.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/controller/MobileModuleController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.mobile.modular.resource.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -32,6 +33,7 @@ import vip.xiaonuo.mobile.modular.resource.param.module.MobileModuleIdParam; import vip.xiaonuo.mobile.modular.resource.param.module.MobileModulePageParam; import vip.xiaonuo.mobile.modular.resource.service.MobileModuleService; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +43,7 @@ import java.util.List; * @date 2022/6/27 14:12 **/ @Tag(name = "移动端模块控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 6) @RestController @Validated public class MobileModuleController { @@ -54,6 +57,7 @@ public class MobileModuleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取移动端模块分页") @GetMapping("/mobile/module/page") public CommonResult> page(MobileModulePageParam mobileModulePageParam) { @@ -66,6 +70,7 @@ public class MobileModuleController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加移动端模块") @CommonLog("添加移动端模块") @PostMapping("/mobile/module/add") @@ -80,6 +85,7 @@ public class MobileModuleController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑移动端模块") @CommonLog("编辑移动端模块") @PostMapping("/mobile/module/edit") @@ -94,11 +100,12 @@ public class MobileModuleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除移动端模块") @CommonLog("删除移动端模块") @PostMapping("/mobile/module/delete") public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List mobileModuleIdParamList) { + List mobileModuleIdParamList) { mobileModuleService.delete(mobileModuleIdParamList); return CommonResult.ok(); } @@ -109,6 +116,7 @@ public class MobileModuleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取移动端模块详情") @GetMapping("/mobile/module/detail") public CommonResult detail(@Valid MobileModuleIdParam mobileModuleIdParam) { diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileButton.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileButton.java index 6f34bb23..25e9a5a2 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileButton.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileButton.java @@ -57,6 +57,6 @@ public class MobileButton extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileMenu.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileMenu.java index 9d4e79c1..f9dd007b 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileMenu.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileMenu.java @@ -87,6 +87,6 @@ public class MobileMenu extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileModule.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileModule.java index 209ed6dd..a774218b 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileModule.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/entity/MobileModule.java @@ -61,6 +61,6 @@ public class MobileModule extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileButtonApiProvider.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileButtonApiProvider.java index c8453a17..9c835219 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileButtonApiProvider.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileButtonApiProvider.java @@ -34,7 +34,7 @@ public class MobileButtonApiProvider implements MobileButtonApi { private MobileButtonService mobileButtonService; @Override - public List listByIds(List buttonIdList) { + public List listButtonCodeListByIdList(List buttonIdList) { return mobileButtonService.listByIds(buttonIdList).stream().map(MobileButton::getCode).collect(Collectors.toList()); } } diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileMenuApiProvider.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileMenuApiProvider.java index bd9a1c3a..47e0ff03 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileMenuApiProvider.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/provider/MobileMenuApiProvider.java @@ -12,11 +12,13 @@ */ package vip.xiaonuo.mobile.modular.resource.provider; +import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.lang.tree.Tree; import cn.hutool.json.JSONObject; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import vip.xiaonuo.mobile.api.MobileMenuApi; +import vip.xiaonuo.mobile.modular.resource.entity.MobileMenu; import vip.xiaonuo.mobile.modular.resource.service.MobileMenuService; import java.util.List; @@ -38,6 +40,11 @@ public class MobileMenuApiProvider implements MobileMenuApi { return mobileMenuService.mobileMenuTreeSelector(); } + @Override + public List mobileMenuTreeSelector(List originDataList) { + return mobileMenuService.mobileMenuTreeSelector(BeanUtil.copyToList(originDataList, MobileMenu.class)); + } + @Override public List> loginMobileMenuTree(List menuIdList) { return mobileMenuService.loginMobileMenuTree(menuIdList); diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/MobileMenuService.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/MobileMenuService.java index 6edd3e94..963b4b01 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/MobileMenuService.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/MobileMenuService.java @@ -127,6 +127,14 @@ public interface MobileMenuService extends IService { **/ List mobileMenuTreeSelector(); + /** + * 获取移动端菜单授权树 + * + * @author xuyuxiang + * @date 2023/1/31 10:14 + **/ + List mobileMenuTreeSelector(List originDataList); + /** * 获取移动端登录菜单树 * diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileButtonServiceImpl.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileButtonServiceImpl.java index abcf40a6..df04dce3 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileButtonServiceImpl.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileButtonServiceImpl.java @@ -22,6 +22,7 @@ import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.page.CommonPageRequest; @@ -102,6 +103,7 @@ public class MobileButtonServiceImpl extends ServiceImpl mobileButtonIdParamList) { List buttonIdList = CollStreamUtil.toList(mobileButtonIdParamList, MobileButtonIdParam::getId); diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileMenuServiceImpl.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileMenuServiceImpl.java index 73fd7204..131f18f8 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileMenuServiceImpl.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileMenuServiceImpl.java @@ -24,6 +24,7 @@ import cn.hutool.core.util.StrUtil; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; @@ -61,16 +62,16 @@ public class MobileMenuServiceImpl extends ServiceImpl> tree(MobileMenuTreeParam mobileMenuTreeParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); if(ObjectUtil.isNotEmpty(mobileMenuTreeParam.getModule())) { - lambdaQueryWrapper.eq(MobileMenu::getModule, mobileMenuTreeParam.getModule()); + queryWrapper.lambda().eq(MobileMenu::getModule, mobileMenuTreeParam.getModule()); } if(ObjectUtil.isNotEmpty(mobileMenuTreeParam.getSearchKey())) { - lambdaQueryWrapper.like(MobileMenu::getTitle, mobileMenuTreeParam.getSearchKey()); + queryWrapper.lambda().like(MobileMenu::getTitle, mobileMenuTreeParam.getSearchKey()); } - lambdaQueryWrapper.eq(MobileMenu::getCategory, MobileResourceCategoryEnum.MENU.getValue()); - lambdaQueryWrapper.orderByDesc(MobileMenu::getSortCode); - List mobileMenuList = this.list(lambdaQueryWrapper); + queryWrapper.lambda().eq(MobileMenu::getCategory, MobileResourceCategoryEnum.MENU.getValue()); + queryWrapper.lambda().orderByDesc(MobileMenu::getSortCode); + List mobileMenuList = this.list(queryWrapper.lambda()); List> treeNodeList = mobileMenuList.stream().map(mobileMenu -> new TreeNode<>(mobileMenu.getId(), mobileMenu.getParentId(), mobileMenu.getTitle(), mobileMenu.getSortCode()).setExtra(JSONUtil.parseObj(mobileMenu))) @@ -98,6 +99,7 @@ public class MobileMenuServiceImpl extends ServiceImpl originDataList = this.list(new LambdaQueryWrapper().eq(MobileMenu::getCategory, MobileResourceCategoryEnum.MENU.getValue())); boolean errorLevel = this.getChildListById(originDataList, mobileMenu.getId(), true).stream() - .map(MobileMenu::getId).collect(Collectors.toList()).contains(mobileMenu.getParentId()); + .map(MobileMenu::getId).toList().contains(mobileMenu.getParentId()); if(errorLevel) { throw new CommonException("不可选择上级菜单:{}", this.getById(originDataList, mobileMenu.getParentId()).getTitle()); } @@ -132,6 +134,7 @@ public class MobileMenuServiceImpl extends ServiceImpl toDeleteMenuIdList = CollectionUtil.newArrayList(); mobileMenuIdList.forEach(menuId -> toDeleteMenuIdList.addAll(this.getChildListById(allMenuList, menuId, true).stream() - .map(MobileMenu::getId).collect(Collectors.toList()))); + .map(MobileMenu::getId).toList())); if(ObjectUtil.isNotEmpty(toDeleteMenuIdList)) { // 清除对应的角色与移动端资源信息 sysRelationApi.removeRoleHasMobileMenuRelation(toDeleteMenuIdList); @@ -190,12 +193,12 @@ public class MobileMenuServiceImpl extends ServiceImpl moduleSelector(MobileMenuSelectorModuleParam mobileMenuSelectorModuleParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); if(ObjectUtil.isNotEmpty(mobileMenuSelectorModuleParam.getSearchKey())) { - lambdaQueryWrapper.like(MobileModule::getTitle, mobileMenuSelectorModuleParam.getSearchKey()); + queryWrapper.lambda().like(MobileModule::getTitle, mobileMenuSelectorModuleParam.getSearchKey()); } - lambdaQueryWrapper.eq(MobileModule::getCategory, MobileResourceCategoryEnum.MODULE.getValue()); - return mobileModuleService.list(lambdaQueryWrapper); + queryWrapper.lambda().eq(MobileModule::getCategory, MobileResourceCategoryEnum.MODULE.getValue()); + return mobileModuleService.list(queryWrapper.lambda()); } @Override @@ -216,16 +219,19 @@ public class MobileMenuServiceImpl extends ServiceImpl mobileMenuTreeSelector() { - // 模块、菜单、按钮,应该查所有的,当然这里没有已经删除过的 - List allModuleAndMenuList = this.list(); + return this.mobileMenuTreeSelector(this.list()); + } + + @Override + public List mobileMenuTreeSelector(List originDataList) { List mobileModuleList = CollectionUtil.newArrayList(); List mobileMenuList = CollectionUtil.newArrayList(); List mobileButtonList = CollectionUtil.newArrayList(); - if (ObjectUtil.isEmpty(allModuleAndMenuList)) { + if (ObjectUtil.isEmpty(originDataList)) { // 返回空列表 return CollectionUtil.newArrayList(); } - allModuleAndMenuList.forEach(mobileMenu -> { + originDataList.forEach(mobileMenu -> { if (mobileMenu.getCategory().equals(MobileResourceCategoryEnum.MODULE.getValue())) { mobileModuleList.add(mobileMenu); } @@ -242,7 +248,7 @@ public class MobileMenuServiceImpl extends ServiceImpl> treeList = TreeUtil.build(treeNodeList, "0"); mobileMenuList.forEach(mobileMenu -> { - boolean isLeafMenu = this.getChildListById(mobileMenuList, mobileMenu.getId(), false).size() == 0; + boolean isLeafMenu = this.getChildListById(mobileMenuList, mobileMenu.getId(), false).isEmpty(); if(isLeafMenu) { JSONObject mobileRoleGrantResourceMenuResult = JSONUtil.createObj(); BeanUtil.copyProperties(mobileMenu, mobileRoleGrantResourceMenuResult); @@ -306,7 +312,7 @@ public class MobileMenuServiceImpl extends ServiceImpl menuList = allMenuList.stream().filter(mobileMenu -> - menuIdList.contains(mobileMenu.getId())).collect(Collectors.toList()); + menuIdList.contains(mobileMenu.getId())).toList(); // 对获取到的角色对应的菜单列表进行处理,获取父列表 menuList.forEach(mobileMenu -> execRecursionFindParent(allMenuList, mobileMenu.getId(), resultList)); @@ -355,7 +361,7 @@ public class MobileMenuServiceImpl extends ServiceImpl> treeNodeList = resultJsonObjectList.stream().map(jsonObject -> diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileModuleServiceImpl.java b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileModuleServiceImpl.java index 756b2480..ccc3c384 100644 --- a/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileModuleServiceImpl.java +++ b/snowy-plugin/snowy-plugin-mobile/src/main/java/vip/xiaonuo/mobile/modular/resource/service/impl/MobileModuleServiceImpl.java @@ -26,6 +26,7 @@ import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.page.CommonPageRequest; @@ -84,6 +85,9 @@ public class MobileModuleServiceImpl extends ServiceImpl mobileModuleIdParamList) { List mobileModuleIdList = CollStreamUtil.toList(mobileModuleIdParamList, MobileModuleIdParam::getId); diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/COMPLETED.png b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/COMPLETED.png new file mode 100644 index 00000000..1ba0eb11 Binary files /dev/null and b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/COMPLETED.png differ diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/END.png b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/END.png new file mode 100644 index 00000000..7877da5f Binary files /dev/null and b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/END.png differ diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/REJECT.png b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/REJECT.png new file mode 100644 index 00000000..d0d9e28c Binary files /dev/null and b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/REJECT.png differ diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/REVOKE.png b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/REVOKE.png new file mode 100644 index 00000000..dbdc08ac Binary files /dev/null and b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/REVOKE.png differ diff --git a/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/SUSPENDED.png b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/SUSPENDED.png new file mode 100644 index 00000000..5ee4aec4 Binary files /dev/null and b/snowy-plugin/snowy-plugin-mobile/src/main/resources/static/mobile/flw/SUSPENDED.png differ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/enums/SysBuildInEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/enums/SysBuildInEnum.java index 383ca2ac..7e226d43 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/enums/SysBuildInEnum.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/enums/SysBuildInEnum.java @@ -29,8 +29,17 @@ public enum SysBuildInEnum { /** 超管角色编码 */ BUILD_IN_ROLE_CODE("superAdmin", "超管"), - /** 系统内置模块编码 */ - BUILD_IN_MODULE_CODE("system", "系统内置"); + /** 超管职位编码 */ + BUILD_IN_POSITION_CODE("superAdmin", "超管"), + + /** 系统内置系统模块编码 */ + BUILD_IN_MODULE_CODE("system", "系统内置"), + + /** 系统内置业务模块编码 */ + BUILD_IN_MODULE_CODE_BIZ("biz", "系统内置"), + + /** 系统内置非租户菜单编码 */ + BUILD_IN_NO_TEN_MENU_CODE("noten", "非租户菜单"); private final String value; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/enums/SysPasswordComplexityEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/enums/SysPasswordComplexityEnum.java new file mode 100644 index 00000000..15c6197b --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/enums/SysPasswordComplexityEnum.java @@ -0,0 +1,64 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.core.enums; + +import lombok.Getter; + +/** + * 密码复杂度类型枚举 + * + * @author xuyuxiang + * @date 2023/3/3 10:40 + **/ +@Getter +public enum SysPasswordComplexityEnum { + + /** + * 无限制 + */ + REG0("REG0", "无限制"), + + /** + * 必须包含数字和字母 + */ + REG1("REG1", "必须包含数字和字母"), + + /** + * 必须包含数字和大写字母 + */ + REG2("REG2", "必须包含数字和大写字母"), + + /** + * 必须包含数字、大写字母、小写字母和特殊字符 + */ + REG3("REG3", "必须包含数字、大写字母、小写字母和特殊字符"), + + /** + * 至少包含数字、字母和特殊字符中的两种 + */ + REG4("REG4", "至少包含数字、字母和特殊字符中的两种"), + + /** + * 至少包含数字、大写字母、小写字母和特殊字符的三种 + */ + REG5("REG5", "至少包含数字、大写字母、小写字母和特殊字符的三种"); + + private final String value; + + private final String message; + + SysPasswordComplexityEnum(String value, String message) { + this.value = value; + this.message = message; + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/listener/SysDataChangeListener.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/listener/SysDataChangeListener.java index fccd3bfc..1555f24c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/listener/SysDataChangeListener.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/listener/SysDataChangeListener.java @@ -15,15 +15,11 @@ package vip.xiaonuo.sys.core.listener; import cn.dev33.satoken.stp.StpUtil; import cn.hutool.json.JSONArray; import cn.hutool.json.JSONObject; -import jakarta.annotation.Resource; import org.springframework.stereotype.Component; import vip.xiaonuo.auth.core.pojo.SaBaseLoginUser; import vip.xiaonuo.auth.core.util.StpLoginUserUtil; -import vip.xiaonuo.common.cache.CommonCacheOperator; import vip.xiaonuo.common.listener.CommonDataChangeListener; import vip.xiaonuo.sys.core.enums.SysDataTypeEnum; -import vip.xiaonuo.sys.modular.org.service.impl.SysOrgServiceImpl; -import vip.xiaonuo.sys.modular.user.service.impl.SysUserServiceImpl; import java.util.List; @@ -36,9 +32,6 @@ import java.util.List; @Component public class SysDataChangeListener implements CommonDataChangeListener { - @Resource - private CommonCacheOperator commonCacheOperator; - @Override public void doAddWithDataId(String dataType, String dataId) { // 此处可做额外处理 @@ -46,20 +39,14 @@ public class SysDataChangeListener implements CommonDataChangeListener { @Override public void doAddWithDataIdList(String dataType, List dataIdList) { - // 如果检测到机构增加,则将机构的数据缓存清除 + // 如果检测到机构增加,则将该机构加入到当前登录用户的数据范围缓存 if(dataType.equals(SysDataTypeEnum.ORG.getValue())) { - commonCacheOperator.remove(SysOrgServiceImpl.ORG_CACHE_ALL_KEY); - // 并将该机构加入到当前登录用户的数据范围缓存 SaBaseLoginUser saBaseLoginUser = StpLoginUserUtil.getLoginUser(); saBaseLoginUser.getDataScopeList().forEach(dataScope -> dataScope.getDataScope().addAll(dataIdList)); saBaseLoginUser.setDataScopeList(saBaseLoginUser.getDataScopeList()); // 重新缓存当前登录用户信息 StpUtil.getTokenSession().set("loginUser", saBaseLoginUser); } - // 如果检测到用户增加,则将用户数据缓存清除 - if(dataType.equals(SysDataTypeEnum.USER.getValue())) { - commonCacheOperator.remove(SysUserServiceImpl.USER_CACHE_ALL_KEY); - } } @Override @@ -79,14 +66,7 @@ public class SysDataChangeListener implements CommonDataChangeListener { @Override public void doUpdateWithDataIdList(String dataType, List dataIdList) { - // 如果检测到机构更新,则将机构的数据缓存清除 - if(dataType.equals(SysDataTypeEnum.ORG.getValue())) { - commonCacheOperator.remove(SysOrgServiceImpl.ORG_CACHE_ALL_KEY); - } - // 如果检测到用户更新,则将用户数据缓存清除 - if(dataType.equals(SysDataTypeEnum.USER.getValue())) { - commonCacheOperator.remove(SysUserServiceImpl.USER_CACHE_ALL_KEY); - } + // 此处可做额外处理 } @Override @@ -106,14 +86,6 @@ public class SysDataChangeListener implements CommonDataChangeListener { @Override public void doDeleteWithDataIdList(String dataType, List dataIdList) { - // 如果检测到机构增加,则将机构的数据缓存清除 - if(dataType.equals(SysDataTypeEnum.ORG.getValue())) { - commonCacheOperator.remove(SysOrgServiceImpl.ORG_CACHE_ALL_KEY); - } - // 如果检测到用户删除,则将用户数据缓存清除,并将这些用户踢下线 - if(dataType.equals(SysDataTypeEnum.USER.getValue())) { - commonCacheOperator.remove(SysUserServiceImpl.USER_CACHE_ALL_KEY); - dataIdList.forEach(StpUtil::kickout); - } + // 此处可做额外处理 } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/timer/SysUserPasswordExpiredNoticeTimerTaskRunner.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/timer/SysUserPasswordExpiredNoticeTimerTaskRunner.java new file mode 100644 index 00000000..0c66d7d1 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/timer/SysUserPasswordExpiredNoticeTimerTaskRunner.java @@ -0,0 +1,39 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.core.timer; + +import jakarta.annotation.Resource; +import lombok.extern.slf4j.Slf4j; +import org.springframework.stereotype.Component; +import vip.xiaonuo.common.timer.CommonTimerTaskRunner; +import vip.xiaonuo.sys.modular.user.service.SysUserService; + +/** + * 通知用户密码即将到期定时类 + * + * @author xuyuxiang + * @date 2022/8/5 15:52 + **/ +@Slf4j +@Component +public class SysUserPasswordExpiredNoticeTimerTaskRunner implements CommonTimerTaskRunner { + + @Resource + private SysUserService sysUserService; + + @Override + public void action(String extJson) { + // 通知用户密码即将到期 + sysUserService.noticeUserPasswordAboutToExpired(); + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/util/SysEmailFormatUtl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/util/SysEmailFormatUtl.java new file mode 100644 index 00000000..f063ad93 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/util/SysEmailFormatUtl.java @@ -0,0 +1,48 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.core.util; + +import cn.hutool.core.date.DateUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONObject; +import vip.xiaonuo.dev.api.DevConfigApi; + +/** + * 系统相关邮件格式化工具类 + * + * @author xuyuxiang + * @date 2025/3/21 19:07 + **/ +public class SysEmailFormatUtl { + + /** 系统名称 */ + private static final String SNOWY_SYS_NAME_KEY = "SNOWY_SYS_NAME"; + + /** + * 格式化邮件内容 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static String format(String content, JSONObject paramMap) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 获取系统名称 + String sysName = devConfigApi.getValueByKey(SNOWY_SYS_NAME_KEY); + // 系统名称 + paramMap.set("sysName", sysName); + // 当前时间 + paramMap.set("sysNowTime", DateUtil.now()); + return StrUtil.format(content, paramMap); + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/util/SysPasswordUtl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/util/SysPasswordUtl.java new file mode 100644 index 00000000..e52632ad --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/core/util/SysPasswordUtl.java @@ -0,0 +1,460 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.core.util; + +import cn.hutool.core.convert.Convert; +import cn.hutool.core.date.DateTime; +import cn.hutool.core.date.DateUnit; +import cn.hutool.core.date.DateUtil; +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.extra.pinyin.PinyinUtil; +import cn.hutool.extra.spring.SpringUtil; +import cn.hutool.json.JSONObject; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import vip.xiaonuo.common.exception.CommonException; +import vip.xiaonuo.common.util.CommonCryptogramUtil; +import vip.xiaonuo.dev.api.DevConfigApi; +import vip.xiaonuo.dev.api.DevWeakPasswordApi; +import vip.xiaonuo.sys.core.enums.SysPasswordComplexityEnum; +import vip.xiaonuo.sys.modular.user.entity.SysUser; +import vip.xiaonuo.sys.modular.user.entity.SysUserExt; +import vip.xiaonuo.sys.modular.user.result.SysLoginUser; +import vip.xiaonuo.sys.modular.user.service.SysUserExtService; +import vip.xiaonuo.sys.modular.user.service.SysUserPasswordService; +import vip.xiaonuo.sys.modular.user.service.SysUserService; + +import java.util.ArrayList; +import java.util.List; +import java.util.stream.Collectors; + +/** + * 系统密码工具类 + * + * @author xuyuxiang + * @date 2025/3/21 19:07 + **/ +public class SysPasswordUtl { + + /** B端系统默认密码 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_FOR_B"; + + /** B端系统密码修改验证方式 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_B"; + + /** B端密码最小长度 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_B"; + + /** B端密码最大长度 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_B"; + + /** B端密码复杂度 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_B"; + + /** B端密码不能连续存在相同字符个数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_B_KEY= "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_B"; + + /** B端密码不能包含用户信息开关(开启后,密码中将不能包含账号、手机号、邮箱前缀和姓名拼音) */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_B"; + + /** B端密码不能使用历史密码开关 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_B"; + + /** B端密码不能使用历史密码范围个数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_B"; + + /** B端密码不能使用弱密码库中密码开关 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_B"; + + /** B端密码自定义额外弱密码库 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_B"; + + /** B端密码有效期天数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_B"; + + /** B端密码过期前提醒天数 */ + private static final String SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_B_KEY = "SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_B"; + + /** + * 获取系统默认密码 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static String getDefaultPassword() { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + String defaultPassword = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_FOR_B_KEY); + if(ObjectUtil.isEmpty(defaultPassword)){ + throw new CommonException("请联系管理员配置系统默认密码"); + } + return defaultPassword; + } + + /** + * 获取修改密码验证方式及配置 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static JSONObject getUpdatePasswordValidConfig() { + JSONObject jsonObject = new JSONObject(); + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 修改密码验证方式 + String passwordUpdateValidType = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_B_KEY); + if(ObjectUtil.isEmpty(passwordUpdateValidType)){ + throw new CommonException("请联系管理员配置系统密码修改验证方式"); + } + jsonObject.set(SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_B_KEY, passwordUpdateValidType); + // 密码最小长度 + String passwordUpdateMinLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_B_KEY); + if(ObjectUtil.isEmpty(passwordUpdateMinLength)){ + throw new CommonException("请联系管理员配置系统密码修改最小长度"); + } + jsonObject.set(SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_B_KEY, passwordUpdateMinLength); + // 密码最大长度 + String passwordUpdateMaxLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_B_KEY); + if(ObjectUtil.isEmpty(passwordUpdateMaxLength)){ + throw new CommonException("请联系管理员配置系统密码修改密码最大长度"); + } + jsonObject.set(SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_B_KEY, passwordUpdateMaxLength); + // 密码复杂度 + String passwordUpdateComplexity = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_B_KEY); + if(ObjectUtil.isEmpty(passwordUpdateValidType)){ + throw new CommonException("请联系管理员配置系统密码修改密码复杂度"); + } + jsonObject.set(SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_B_KEY, passwordUpdateComplexity); + // 不能连续出现相同字符个数 + String passwordUpdateNotAllowContinuousSame = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_B_KEY); + if(ObjectUtil.isEmpty(passwordUpdateValidType)){ + throw new CommonException("请联系管理员配置系统密码修改联系出现相同字符的个数"); + } + jsonObject.set(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_B_KEY, passwordUpdateNotAllowContinuousSame); + return jsonObject; + } + + /** + * 获取修改密码验证方式 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static String getUpdatePasswordValidType() { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + String passwordUpdateValidType = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_B_KEY); + if(ObjectUtil.isEmpty(passwordUpdateValidType)){ + throw new CommonException("请联系管理员配置系统密码修改验证方式"); + } + return passwordUpdateValidType; + } + + /** + * 校验修改密码验证方式 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validUpdatePasswordValidType(String type) { + String updatePasswordValidType = getUpdatePasswordValidType(); + if(!updatePasswordValidType.equals(type)){ + throw new CommonException("系统配置不支持此方式修改密码"); + } + } + + /** + * 校验新密码 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(SysLoginUser sysLoginUser, String newPassword) { + String userId = sysLoginUser.getId(); + String account = sysLoginUser.getAccount(); + String name = sysLoginUser.getName(); + String phone = sysLoginUser.getPhone(); + String email = sysLoginUser.getEmail(); + validNewPassword(newPassword); + validNewPassword(userId, account, name, phone, email, newPassword); + } + + /** + * 校验新密码 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(SysUser sysUser, String newPassword) { + String userId = sysUser.getId(); + String account = sysUser.getAccount(); + String name = sysUser.getName(); + String phone = sysUser.getPhone(); + String email = sysUser.getEmail(); + validNewPassword(newPassword); + validNewPassword(userId, account, name, phone, email, newPassword); + } + + /** + * 校验新密码(基本要求) + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(String newPassword) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 密码最小长度 + String passwordMinLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordMinLength)){ + Integer minLengthInt = Convert.toInt(passwordMinLength); + if(newPassword.length() < minLengthInt){ + throw new CommonException("密码最小长度应为:{}", minLengthInt); + } + } + // 密码最大长度 + String passwordMaxLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordMaxLength)){ + Integer maxLengthInt = Convert.toInt(passwordMaxLength); + if(newPassword.length() > maxLengthInt){ + throw new CommonException("密码最大长度应为:{}", maxLengthInt); + } + } + // 密码复杂度 + String passwordComplexity = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordComplexity)){ + // 据复杂度配置判断 + boolean notLimit = SysPasswordComplexityEnum.REG0.getValue().equals(passwordComplexity); + if(!notLimit){ + if(SysPasswordComplexityEnum.REG1.getValue().equals(passwordComplexity)) { + String reg = "^(?=.*[0-9])(?=.*[a-zA-Z]).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", SysPasswordComplexityEnum.REG1.getMessage()); + } + } + if(SysPasswordComplexityEnum.REG2.getValue().equals(passwordComplexity)) { + String reg = "^(?=.*[0-9])(?=.*[A-Z]).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", SysPasswordComplexityEnum.REG2.getMessage()); + } + } + if(SysPasswordComplexityEnum.REG3.getValue().equals(passwordComplexity)) { + String reg = "^(?=.*[0-9])(?=.*[A-Z])(?=.*[a-z])(?=.*[^a-zA-Z0-9]).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", SysPasswordComplexityEnum.REG3.getMessage()); + } + } + if(SysPasswordComplexityEnum.REG4.getValue().equals(passwordComplexity)) { + String reg = "^(?:(?=.*[0-9])(?=.*[a-zA-Z])|(?=.*[0-9])(?=.*[^a-zA-Z0-9])|(?=.*[a-zA-Z])(?=.*[^a-zA-Z0-9])).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", SysPasswordComplexityEnum.REG4.getMessage()); + } + } + if(SysPasswordComplexityEnum.REG5.getValue().equals(passwordComplexity)) { + String reg = "^(?:(?=.*[0-9])(?=.*[A-Z])(?=.*[a-z])|(?=.*[0-9])(?=.*[A-Z])(?=.*[^a-zA-Z0-9])|(?=.*[0-9])(?=.*[a-z])(?=.*[^a-zA-Z0-9])|(?=.*[A-Z])(?=.*[a-z])(?=.*[^a-zA-Z0-9])).+$"; + if(!newPassword.matches(reg)) { + throw new CommonException("密码{}", SysPasswordComplexityEnum.REG5.getMessage()); + } + } + } + } + // 密码不能连续存在相同字符个数 + String passwordNotAllowContinuousSameCharacterLength = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowContinuousSameCharacterLength)){ + // 密码不能连续存在相同字符个数 + Integer passwordNotAllowContinuousSameCharacterLengthInt = Convert.toInt(passwordNotAllowContinuousSameCharacterLength); + boolean hasConsecutiveChars = hasConsecutiveChars(newPassword, passwordNotAllowContinuousSameCharacterLengthInt); + if(hasConsecutiveChars){ + throw new CommonException("密码不可存在" + passwordNotAllowContinuousSameCharacterLengthInt + "个连续相同的字符"); + } + } + // 密码不能使用弱密码库中密码开关 + String passwordNotAllowUseWeakFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowUseWeakFlag)){ + if(Convert.toBool(passwordNotAllowUseWeakFlag)){ + // 不能包含内置弱密码和自定义额外弱密码库中密码 + DevWeakPasswordApi devWeakPasswordApi = SpringUtil.getBean(DevWeakPasswordApi.class); + List weakPasswordList = devWeakPasswordApi.weakPasswordList(); + // 获取自定义额外弱密码库 + String passwordDefineWeakDatabase = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordDefineWeakDatabase)){ + weakPasswordList.addAll(StrUtil.split(passwordDefineWeakDatabase, StrUtil.COMMA)); + } + if(weakPasswordList.contains(newPassword)){ + throw new CommonException("密码不可使用弱密码"); + } + } + } + } + + /** + * 校验新密码(含用户信息等) + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static void validNewPassword(String userId, String account, String name, String phone, String email, String newPassword) { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 密码不能包含用户信息开关 + String passwordNotAllowContainsUserInfoFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowContainsUserInfoFlag)){ + if(Convert.toBool(passwordNotAllowContainsUserInfoFlag)){ + // 账号,不能包含 + if(ObjectUtil.isNotEmpty(account) && newPassword.contains(account)){ + throw new CommonException("密码不可包含账号"); + } + // 姓名,不能包含拼音 + if(ObjectUtil.isNotEmpty(name) && newPassword.contains(PinyinUtil.getPinyin(name, ""))){ + throw new CommonException("密码不可包含姓名拼音"); + } + // 手机号,不能包含 + if(ObjectUtil.isNotEmpty(phone) && newPassword.contains(phone)){ + throw new CommonException("密码不可包含手机号"); + } + // 邮箱,不能包含前缀,即@字符之前内容 + if(ObjectUtil.isNotEmpty(email) && newPassword.contains(StrUtil.split(email, StrUtil.AT).get(0))){ + throw new CommonException("密码不可包含邮箱前缀"); + } + } + } + // 密码不能使用历史密码开关 + String passwordNotAllowUseHistoryFlag = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowUseHistoryFlag)){ + if(Convert.toBool(passwordNotAllowUseHistoryFlag)){ + // 密码不能使用历史密码范围个数 + String passwordNotAllowContainsHistoryCount = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(passwordNotAllowContainsHistoryCount)){ + Integer passwordNotAllowContainsHistoryCountInt = Convert.toInt(passwordNotAllowContainsHistoryCount); + // 取前N次密码,进行判断 + SysUserPasswordService sysUserPasswordService = SpringUtil.getBean(SysUserPasswordService.class); + List userPasswordHistoryLimit = sysUserPasswordService.getUserPasswordHistoryLimit(userId, passwordNotAllowContainsHistoryCountInt); + if(ObjectUtil.isNotEmpty(userPasswordHistoryLimit)){ + if(userPasswordHistoryLimit.contains(CommonCryptogramUtil.doHashValue(newPassword))) { + throw new CommonException("密码不可包含历史密码"); + } + } + } + } + } + } + + /** + * 判断当前用户密码是否过期 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static boolean isUserPasswordExpired(String userId) { + // 先判断是否系统默认密码 + SysUserService sysUserService = SpringUtil.getBean(SysUserService.class); + SysUser sysUser = sysUserService.queryEntity(userId); + String defaultPassword = getDefaultPassword(); + // 若是,则认为已过期,需要修改 + if(CommonCryptogramUtil.doHashValue(defaultPassword).equals(sysUser.getPassword())){ + return true; + } + SysUserExtService sysUserExtService = SpringUtil.getBean(SysUserExtService.class); + // 获取用户扩展信息 + SysUserExt sysUserExt = sysUserExtService.getOne(new LambdaQueryWrapper().eq(SysUserExt::getUserId, userId)); + if(ObjectUtil.isEmpty(sysUserExt)){ + // 无扩展信息的,直接认定为密码过期 + return true; + } else { + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 获取密码有效期天数 + String passwordExpiredDays = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_B_KEY); + if(ObjectUtil.isEmpty(passwordExpiredDays)){ + throw new CommonException("请联系管理员配置密码有效期天数"); + } else { + // 判断上次密码修改时间距今天的天数 + long betweenDays = DateUtil.between(sysUserExt.getPasswordUpdateTime(), DateTime.now(), DateUnit.DAY, false); + // 如果距离今天已经超过了系统配置的有效期,则认为已过期 + return betweenDays >= Convert.toInt(passwordExpiredDays); + } + } + } + + /** + * 获取今日需要提醒密码到期的用户集合 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + public static List thisDayPasswordExpiredNeedNoticeUserIdList() { + List resultUserList = new ArrayList<>(); + DevConfigApi devConfigApi = SpringUtil.getBean(DevConfigApi.class); + // 获取密码有效期天数 + String passwordExpiredDays = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_B_KEY); + // 密码有效期配置如果不为空 + if(ObjectUtil.isNotEmpty(passwordExpiredDays)){ + // 获取密码过期前提醒天数 + String passwordExpiredNoticeDays = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_B_KEY); + // 密码过期前提醒天数如果不为空 + if(ObjectUtil.isNotEmpty(passwordExpiredDays)){ + // 获取密码修改日期距离提醒日的天数N + int noticeOffsetDays = Convert.toInt(passwordExpiredDays) - Convert.toInt(passwordExpiredNoticeDays); + // 提前提醒天数必须大于过期天数 + if(noticeOffsetDays > 0) { + // 今日需要提醒的,即密码修改日期为N天前的 + DateTime passWordUpdateTime = DateUtil.offsetDay(DateTime.now(), -noticeOffsetDays); + // 获取当天的开始时间 + DateTime beginDay = DateUtil.beginOfDay(passWordUpdateTime); + // 获取当天的结束时间 + DateTime endDay = DateUtil.endOfDay(passWordUpdateTime); + // 获取SysUserExtService + SysUserExtService sysUserExtService = SpringUtil.getBean(SysUserExtService.class); + // 查询密码修改日期在该范围内的(无修改密码历史(即扩展信息为空)的无需提醒,会强制修改密码) + List userIdList = sysUserExtService.list(new LambdaQueryWrapper() + .between(SysUserExt::getPasswordUpdateTime,beginDay, endDay)).stream() + .map(SysUserExt::getUserId).collect(Collectors.toList()); + if(ObjectUtil.isNotEmpty(userIdList)){ + // 获取SysUserXService + SysUserService sysUserService = SpringUtil.getBean(SysUserService.class); + resultUserList = sysUserService.listByIds(userIdList); + } + } + } + } + return resultUserList; + } + + /** + * 判断字符串是否存在N个连续相同的字符 + * + * @author xuyuxiang + * @date 2025/3/21 19:08 + **/ + private static boolean hasConsecutiveChars(String str, int n) { + // 处理无效输入:字符串为空、n非正、或字符串长度不足 + if (str == null || n <= 0 || str.length() < n) { + return false; + } + // 特殊情况:n=1时只要字符串非空即存在连续1个字符 + if (n == 1) { + return true; + } + + int maxStartIndex = str.length() - n; + for (int i = 0; i <= maxStartIndex; i++) { + char currentChar = str.charAt(i); + boolean isConsecutive = true; + // 检查后续n-1个字符是否相同 + for (int j = 1; j < n; j++) { + if (str.charAt(i + j) != currentChar) { + isConsecutive = false; + break; + } + } + if (isConsecutive) { + return true; + } + } + return false; + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/controller/SysGroupController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/controller/SysGroupController.java index ab1411e6..0291948d 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/controller/SysGroupController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/controller/SysGroupController.java @@ -14,8 +14,11 @@ package vip.xiaonuo.sys.modular.group.controller; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; -import io.swagger.v3.oas.annotations.tags.Tag; import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.annotation.Resource; +import jakarta.validation.Valid; +import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; @@ -26,9 +29,6 @@ import vip.xiaonuo.common.pojo.CommonResult; import vip.xiaonuo.sys.modular.group.entity.SysGroup; import vip.xiaonuo.sys.modular.group.param.*; import vip.xiaonuo.sys.modular.group.service.SysGroupService; -import jakarta.annotation.Resource; -import jakarta.validation.Valid; -import jakarta.validation.constraints.NotEmpty; import vip.xiaonuo.sys.modular.user.entity.SysUser; import java.util.List; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/entity/SysGroup.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/entity/SysGroup.java index 60bd0a90..87537a5d 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/entity/SysGroup.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/entity/SysGroup.java @@ -12,7 +12,8 @@ */ package vip.xiaonuo.sys.modular.group.entity; -import com.baomidou.mybatisplus.annotation.*; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; import io.swagger.v3.oas.annotations.media.Schema; import lombok.Getter; import lombok.Setter; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/mapper/mapping/SysGroupMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/mapper/mapping/SysGroupMapper.xml index 4534cb42..9d73ad02 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/mapper/mapping/SysGroupMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/mapper/mapping/SysGroupMapper.xml @@ -2,4 +2,4 @@ - \ No newline at end of file + diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupAddParam.java index 354a8ab9..1a2eb1d0 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupAddParam.java @@ -13,10 +13,10 @@ package vip.xiaonuo.sys.modular.group.param; import io.swagger.v3.oas.annotations.media.Schema; -import lombok.Getter; -import lombok.Setter; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; +import lombok.Getter; +import lombok.Setter; /** * 用户组添加参数 diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupEditParam.java index efdc7bdd..95bb1d85 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupEditParam.java @@ -13,10 +13,10 @@ package vip.xiaonuo.sys.modular.group.param; import io.swagger.v3.oas.annotations.media.Schema; -import lombok.Getter; -import lombok.Setter; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; +import lombok.Getter; +import lombok.Setter; /** * 用户组编辑参数 diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupIdParam.java index 1ffbcc69..ac45b94b 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/param/SysGroupIdParam.java @@ -13,9 +13,9 @@ package vip.xiaonuo.sys.modular.group.param; import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; import lombok.Getter; import lombok.Setter; -import jakarta.validation.constraints.NotBlank; /** * 用户组Id参数 diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/provider/SysGroupApiProvider.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/provider/SysGroupApiProvider.java index 7cfaed6b..7e4384ea 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/provider/SysGroupApiProvider.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/provider/SysGroupApiProvider.java @@ -22,6 +22,7 @@ import vip.xiaonuo.sys.modular.group.param.SysGroupGrantUserParam; import vip.xiaonuo.sys.modular.group.param.SysGroupIdParam; import vip.xiaonuo.sys.modular.group.param.SysGroupSelectorParam; import vip.xiaonuo.sys.modular.group.service.SysGroupService; + import java.util.List; /** diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/service/impl/SysGroupServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/service/impl/SysGroupServiceImpl.java index 43194d38..cbdb5d8f 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/service/impl/SysGroupServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/group/service/impl/SysGroupServiceImpl.java @@ -39,6 +39,7 @@ import vip.xiaonuo.sys.modular.relation.entity.SysRelation; import vip.xiaonuo.sys.modular.relation.enums.SysRelationCategoryEnum; import vip.xiaonuo.sys.modular.relation.service.SysRelationService; import vip.xiaonuo.sys.modular.user.entity.SysUser; +import vip.xiaonuo.sys.modular.user.enums.SysUserStatusEnum; import vip.xiaonuo.sys.modular.user.service.SysUserService; import java.util.List; @@ -131,9 +132,11 @@ public class SysGroupServiceImpl extends ServiceImpl i @Override public Page userSelector(SysGroupSelectorUserParam sysGroupSelectorUserParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + // 只查询状态为正常的 + queryWrapper.lambda().eq(SysUser::getUserStatus, SysUserStatusEnum.ENABLE.getValue()); // 只查询部分字段 - lambdaQueryWrapper.select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, + queryWrapper.lambda().select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, SysUser::getName, SysUser::getSortCode, SysUser::getGender, SysUser::getEntryDate); // 如果查询条件为空,则直接查询 if(ObjectUtil.isAllEmpty(sysGroupSelectorUserParam.getOrgId(), sysGroupSelectorUserParam.getSearchKey())) { @@ -144,16 +147,16 @@ public class SysGroupServiceImpl extends ServiceImpl i List childOrgIdList = CollStreamUtil.toList(sysOrgService.getChildListById(sysOrgService .getAllOrgList(), sysGroupSelectorUserParam.getOrgId(), true), SysOrg::getId); if (ObjectUtil.isNotEmpty(childOrgIdList)) { - lambdaQueryWrapper.in(SysUser::getOrgId, childOrgIdList); + queryWrapper.lambda().in(SysUser::getOrgId, childOrgIdList); } else { return new Page<>(); } } if (ObjectUtil.isNotEmpty(sysGroupSelectorUserParam.getSearchKey())) { - lambdaQueryWrapper.like(SysUser::getName, sysGroupSelectorUserParam.getSearchKey()); + queryWrapper.lambda().like(SysUser::getName, sysGroupSelectorUserParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysUser::getSortCode); - return sysUserService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysUser::getSortCode); + return sysUserService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } @@ -174,16 +177,16 @@ public class SysGroupServiceImpl extends ServiceImpl i @Override public Page groupSelector(SysGroupSelectorParam sysGroupSelectorParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 只查询部分字段,排掉extJson - lambdaQueryWrapper.select(SysGroup::getId, SysGroup::getName, SysGroup::getSortCode, SysGroup::getRemark); - lambdaQueryWrapper.orderByAsc(SysGroup::getSortCode); + queryWrapper.lambda().select(SysGroup::getId, SysGroup::getName, SysGroup::getSortCode, SysGroup::getRemark); + queryWrapper.lambda().orderByAsc(SysGroup::getSortCode); // 如果查询条件为空,则直接查询 if (!ObjectUtil.isAllEmpty(sysGroupSelectorParam.getSearchKey())) { if (ObjectUtil.isNotEmpty(sysGroupSelectorParam.getSearchKey())) { - lambdaQueryWrapper.like(SysGroup::getName, sysGroupSelectorParam.getSearchKey()); + queryWrapper.lambda().like(SysGroup::getName, sysGroupSelectorParam.getSearchKey()); } } - return this.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + return this.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/controller/SysIndexController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/controller/SysIndexController.java index c020405f..530b58ec 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/controller/SysIndexController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/controller/SysIndexController.java @@ -12,11 +12,11 @@ */ package vip.xiaonuo.sys.modular.index.controller; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -30,6 +30,7 @@ import vip.xiaonuo.sys.modular.index.param.*; import vip.xiaonuo.sys.modular.index.result.*; import vip.xiaonuo.sys.modular.index.service.SysIndexService; +import javax.validation.Valid; import java.util.List; /** @@ -53,6 +54,7 @@ public class SysIndexController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 1) @Operation(summary = "添加日程") @CommonLog("添加日程") @PostMapping("/sys/index/schedule/add") @@ -67,6 +69,7 @@ public class SysIndexController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "删除日程") @CommonLog("删除日程") @PostMapping("/sys/index/schedule/deleteSchedule") @@ -82,6 +85,7 @@ public class SysIndexController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 3) @Operation(summary = "获取日程列表") @GetMapping("/sys/index/schedule/list") public CommonResult> scheduleList(@Valid SysIndexScheduleListParam sysIndexScheduleListParam) { @@ -94,6 +98,7 @@ public class SysIndexController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "获取当前用户站内信列表") @GetMapping("/sys/index/message/list") public CommonResult> messageList(SysIndexMessageListParam sysIndexMessageListParam) { @@ -106,6 +111,7 @@ public class SysIndexController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取站内信详情") @GetMapping("/sys/index/message/detail") public CommonResult messageDetail(@Valid SysIndexMessageIdParam sysIndexMessageIdParam) { @@ -118,6 +124,7 @@ public class SysIndexController { * @author diantu * @date 2023/7/10 */ + @ApiOperationSupport(order = 6) @Operation(summary = "站内信全部标记已读") @PostMapping("/sys/index/message/allMessageMarkRead") public CommonResult allMessageMarkRead() { @@ -131,6 +138,7 @@ public class SysIndexController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取当前用户访问日志列表") @GetMapping("/sys/index/visLog/list") public CommonResult> visLogList() { @@ -143,6 +151,7 @@ public class SysIndexController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 8) @Operation(summary = "获取当前用户操作日志列表") @GetMapping("/sys/index/opLog/list") public CommonResult> opLogList() { @@ -155,6 +164,7 @@ public class SysIndexController { * @author diantu * @date 2023/7/10 **/ + @ApiOperationSupport(order = 9) @Operation(summary = "创建sse连接") @GetMapping("/dev/message/createSseConnect") public SseEmitter createSseConnect(String clientId){ @@ -167,6 +177,7 @@ public class SysIndexController { * @author xuyuxiang、yubaoshan * @date 2024/7/18 17:35 */ + @ApiOperationSupport(order = 10) @Operation(summary = "获取基础系统业务数据") @GetMapping("/sys/index/bizDataCount") public CommonResult getBizDataCount() { @@ -179,6 +190,7 @@ public class SysIndexController { * @author yubaoshan * @date 2024/7/18 17:35 */ + @ApiOperationSupport(order = 11) @Operation(summary = "获取运维一览数据") @GetMapping("/sys/index/opDataCount") public CommonResult getOpDataCount() { @@ -191,6 +203,7 @@ public class SysIndexController { * @author yubaoshan * @date 2024/7/18 17:35 */ + @ApiOperationSupport(order = 12) @Operation(summary = "获取基础工具数据") @GetMapping("/sys/index/toolDataCount") public CommonResult getToolDataCount() { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexMessageIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexMessageIdParam.java index e2f30f47..3e7caaa3 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexMessageIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexMessageIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysIndexMessageIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleAddParam.java index e09f2ae8..5c476ff7 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleAddParam.java @@ -28,17 +28,17 @@ import lombok.Setter; public class SysIndexScheduleAddParam { /** 日程日期 */ - @Schema(description = "日程日期", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "日程日期") @NotBlank(message = "scheduleDate不能为空") private String scheduleDate; /** 日程时间 */ - @Schema(description = "日程时间", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "日程时间") @NotBlank(message = "scheduleTime不能为空") private String scheduleTime; /** 日程内容 */ - @Schema(description = "日程内容", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "日程内容") @NotBlank(message = "scheduleContent不能为空") private String scheduleContent; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleIdParam.java index 90155037..3be14d26 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysIndexScheduleIdParam { /** 日程id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleListParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleListParam.java index bae2da15..02fc2568 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleListParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/param/SysIndexScheduleListParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysIndexScheduleListParam { /** 日程日期 */ - @Schema(description = "日程日期", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "日程日期") @NotBlank(message = "scheduleDate不能为空") private String scheduleDate; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageDetailResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageDetailResult.java index 7f82a14d..50b1f87a 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageDetailResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageDetailResult.java @@ -29,7 +29,7 @@ import java.util.List; public class SysIndexMessageDetailResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 分类 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageListResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageListResult.java index 8da706f8..38f5a4a4 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageListResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexMessageListResult.java @@ -29,7 +29,7 @@ import java.util.Date; public class SysIndexMessageListResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 分类 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexOpLogListResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexOpLogListResult.java index 40609963..da0e37e0 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexOpLogListResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexOpLogListResult.java @@ -31,7 +31,7 @@ import java.util.Date; public class SysIndexOpLogListResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 日志分类 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexScheduleListResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexScheduleListResult.java index 2f92c306..a5fee423 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexScheduleListResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexScheduleListResult.java @@ -27,7 +27,7 @@ import lombok.Setter; public class SysIndexScheduleListResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 用户id */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexVisLogListResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexVisLogListResult.java index 411ecc7c..db6d1537 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexVisLogListResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/index/result/SysIndexVisLogListResult.java @@ -31,7 +31,7 @@ import java.util.Date; public class SysIndexVisLogListResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 日志分类 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/controller/SysOrgController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/controller/SysOrgController.java index cf245c53..8e497606 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/controller/SysOrgController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/controller/SysOrgController.java @@ -14,10 +14,11 @@ package vip.xiaonuo.sys.modular.org.controller; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -27,10 +28,12 @@ import org.springframework.web.bind.annotation.RestController; import vip.xiaonuo.common.annotation.CommonLog; import vip.xiaonuo.common.pojo.CommonResult; import vip.xiaonuo.sys.modular.org.entity.SysOrg; +import vip.xiaonuo.sys.modular.org.enums.SysOrgSourceFromTypeEnum; import vip.xiaonuo.sys.modular.org.param.*; import vip.xiaonuo.sys.modular.org.service.SysOrgService; import vip.xiaonuo.sys.modular.user.entity.SysUser; +import javax.validation.Valid; import java.util.List; /** @@ -40,6 +43,7 @@ import java.util.List; * @date 2022/4/24 19:55 */ @Tag(name = "组织控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 1) @RestController @Validated public class SysOrgController { @@ -53,6 +57,7 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取组织分页") @GetMapping("/sys/org/page") public CommonResult> page(SysOrgPageParam sysOrgPageParam) { @@ -65,6 +70,7 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取组织树") @GetMapping("/sys/org/tree") public CommonResult>> tree() { @@ -77,11 +83,12 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "添加组织") @CommonLog("添加组织") @PostMapping("/sys/org/add") public CommonResult add(@RequestBody @Valid SysOrgAddParam sysOrgAddParam) { - sysOrgService.add(sysOrgAddParam); + sysOrgService.add(sysOrgAddParam, SysOrgSourceFromTypeEnum.SYSTEM_ADD.getValue()); return CommonResult.ok(); } @@ -91,6 +98,7 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "编辑组织") @CommonLog("编辑组织") @PostMapping("/sys/org/edit") @@ -105,6 +113,7 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "删除组织") @CommonLog("删除组织") @PostMapping("/sys/org/delete") @@ -120,6 +129,7 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取组织详情") @GetMapping("/sys/org/detail") public CommonResult detail(@Valid SysOrgIdParam sysOrgIdParam) { @@ -134,6 +144,7 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取组织树选择器") @GetMapping("/sys/org/orgTreeSelector") public CommonResult>> orgTreeSelector() { @@ -146,6 +157,7 @@ public class SysOrgController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 8) @Operation(summary = "获取用户选择器") @GetMapping("/sys/org/userSelector") public CommonResult> userSelector(SysOrgSelectorUserParam sysOrgSelectorUserParam) { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/entity/SysOrg.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/entity/SysOrg.java index 2e3a43c4..a9e4ae44 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/entity/SysOrg.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/entity/SysOrg.java @@ -44,7 +44,7 @@ public class SysOrg extends CommonEntity { /** 主管id */ @Schema(description = "主管id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.SIMPLE, target = SysUser.class, fields = "name", alias = "director", ref = "directorName") private String directorId; @@ -66,6 +66,6 @@ public class SysOrg extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/entity/SysOrgExt.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/entity/SysOrgExt.java new file mode 100644 index 00000000..cc7105b3 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/entity/SysOrgExt.java @@ -0,0 +1,46 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.org.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +/** + * 组织扩展实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("SYS_ORG_EXT") +public class SysOrgExt extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 组织id */ + @Schema(description = "组织id") + private String orgId; + + /** 来源类别 */ + @Schema(description = "来源类别") + private String sourceFromType; + +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/enums/SysOrgSourceFromTypeEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/enums/SysOrgSourceFromTypeEnum.java new file mode 100644 index 00000000..c6cc2ad5 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/enums/SysOrgSourceFromTypeEnum.java @@ -0,0 +1,37 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.org.enums; + +import lombok.Getter; + +/** + * 组织来源类型枚举 + * + * @author xuyuxiang + * @date 2022/4/21 19:56 + **/ +@Getter +public enum SysOrgSourceFromTypeEnum { + + /** 系统自建 */ + SYSTEM_ADD("SYSTEM_ADD"), + + /** 身份源 */ + ID_SOURCE("ID_SOURCE"); + + private final String value; + + SysOrgSourceFromTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/SysOrgExtMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/SysOrgExtMapper.java new file mode 100644 index 00000000..063d717c --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/SysOrgExtMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.org.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.sys.modular.org.entity.SysOrgExt; + +/** + * 组织扩展Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface SysOrgExtMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/SysOrgMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/SysOrgMapper.java index d2a67297..e48863f4 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/SysOrgMapper.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/SysOrgMapper.java @@ -12,10 +12,7 @@ */ package vip.xiaonuo.sys.modular.org.mapper; -import com.baomidou.mybatisplus.annotation.InterceptorIgnore; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import org.apache.ibatis.annotations.Param; import vip.xiaonuo.sys.modular.org.entity.SysOrg; /** @@ -25,13 +22,4 @@ import vip.xiaonuo.sys.modular.org.entity.SysOrg; * @date 2022/4/21 18:37 **/ public interface SysOrgMapper extends BaseMapper { - - /** - * 删除数据并忽略插件(逻辑删除、租户拼接) - * - * @author xuyuxiang - * @date 2023/12/25 23:20 - */ - @InterceptorIgnore(tenantLine = "true") - void deleteIgnoreInterceptor(@Param("ew") LambdaQueryWrapper lambdaQueryWrapper); } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/mapping/SysOrgExtMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/mapping/SysOrgExtMapper.xml new file mode 100644 index 00000000..ae4fc729 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/mapping/SysOrgExtMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/mapping/SysOrgMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/mapping/SysOrgMapper.xml index b76a4280..e9f676e1 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/mapping/SysOrgMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/mapper/mapping/SysOrgMapper.xml @@ -2,7 +2,4 @@ - - DELETE FROM SYS_ORG ${ew.customSqlSegment} - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgAddParam.java index 44b24ec8..28da9195 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgAddParam.java @@ -29,22 +29,22 @@ import lombok.Setter; public class SysOrgAddParam { /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgEditParam.java index e1dba4a9..7fcb6cd1 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgEditParam.java @@ -29,27 +29,27 @@ import lombok.Setter; public class SysOrgEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgIdParam.java index cfb908a9..736d0001 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/param/SysOrgIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysOrgIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/SysOrgExtService.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/SysOrgExtService.java new file mode 100644 index 00000000..c50c705f --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/SysOrgExtService.java @@ -0,0 +1,33 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.org.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.sys.modular.org.entity.SysOrgExt; + +/** + * 组织扩展Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface SysOrgExtService extends IService { + + /** + * 插入扩展信息 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void createExtInfo(String orgId, String sourceFromType); +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/SysOrgService.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/SysOrgService.java index 25a2d03c..fb62b65c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/SysOrgService.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/SysOrgService.java @@ -51,7 +51,7 @@ public interface SysOrgService extends IService { * @author xuyuxiang * @date 2022/4/24 20:48 */ - void add(SysOrgAddParam sysOrgAddParam); + void add(SysOrgAddParam sysOrgAddParam, String sourceFromType); /** * 编辑组织 diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/impl/SysOrgExtServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/impl/SysOrgExtServiceImpl.java new file mode 100644 index 00000000..71cf8721 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/impl/SysOrgExtServiceImpl.java @@ -0,0 +1,37 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.org.service.impl; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import vip.xiaonuo.sys.modular.org.entity.SysOrgExt; +import vip.xiaonuo.sys.modular.org.mapper.SysOrgExtMapper; +import vip.xiaonuo.sys.modular.org.service.SysOrgExtService; + +/** + * 组织扩展Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class SysOrgExtServiceImpl extends ServiceImpl implements SysOrgExtService { + + @Override + public void createExtInfo(String orgId, String sourceFromType) { + SysOrgExt sysOrgExt = new SysOrgExt(); + sysOrgExt.setOrgId(orgId); + sysOrgExt.setSourceFromType(sourceFromType); + this.save(sysOrgExt); + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/impl/SysOrgServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/impl/SysOrgServiceImpl.java index 752868d5..58a7eea4 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/impl/SysOrgServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/org/service/impl/SysOrgServiceImpl.java @@ -30,7 +30,6 @@ import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; -import vip.xiaonuo.common.cache.CommonCacheOperator; import vip.xiaonuo.common.enums.CommonSortOrderEnum; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.listener.CommonDataChangeEventCenter; @@ -38,14 +37,17 @@ import vip.xiaonuo.common.page.CommonPageRequest; import vip.xiaonuo.sys.core.enums.SysDataTypeEnum; import vip.xiaonuo.sys.modular.org.entity.SysOrg; import vip.xiaonuo.sys.modular.org.enums.SysOrgCategoryEnum; +import vip.xiaonuo.sys.modular.org.enums.SysOrgSourceFromTypeEnum; import vip.xiaonuo.sys.modular.org.mapper.SysOrgMapper; import vip.xiaonuo.sys.modular.org.param.*; +import vip.xiaonuo.sys.modular.org.service.SysOrgExtService; import vip.xiaonuo.sys.modular.org.service.SysOrgService; import vip.xiaonuo.sys.modular.position.entity.SysPosition; import vip.xiaonuo.sys.modular.position.service.SysPositionService; import vip.xiaonuo.sys.modular.role.entity.SysRole; import vip.xiaonuo.sys.modular.role.service.SysRoleService; import vip.xiaonuo.sys.modular.user.entity.SysUser; +import vip.xiaonuo.sys.modular.user.enums.SysUserStatusEnum; import vip.xiaonuo.sys.modular.user.service.SysUserService; import java.util.Iterator; @@ -61,10 +63,8 @@ import java.util.stream.Collectors; @Service public class SysOrgServiceImpl extends ServiceImpl implements SysOrgService { - public static final String ORG_CACHE_ALL_KEY = "sys-org:all"; - @Resource - private CommonCacheOperator commonCacheOperator; + private SysOrgExtService sysOrgExtService; @Resource private SysRoleService sysRoleService; @@ -109,7 +109,7 @@ public class SysOrgServiceImpl extends ServiceImpl impleme @Transactional(rollbackFor = Exception.class) @Override - public void add(SysOrgAddParam sysOrgAddParam) { + public void add(SysOrgAddParam sysOrgAddParam, String sourceFromType) { SysOrgCategoryEnum.validate(sysOrgAddParam.getCategory()); SysOrg sysOrg = BeanUtil.toBean(sysOrgAddParam, SysOrg.class); // 重复名称 @@ -119,8 +119,10 @@ public class SysOrgServiceImpl extends ServiceImpl impleme throw new CommonException("存在重复的同级组织,名称为:{}", sysOrg.getName()); } sysOrg.setCode(RandomUtil.randomString(10)); + // 保存组织 this.save(sysOrg); - + // 插入扩展信息 + sysOrgExtService.createExtInfo(sysOrg.getId(), sourceFromType); // 发布增加事件 CommonDataChangeEventCenter.doAddWithData(SysDataTypeEnum.ORG.getValue(), JSONUtil.createArray().put(sysOrg)); } @@ -142,8 +144,8 @@ public class SysOrgServiceImpl extends ServiceImpl impleme if(errorLevel) { throw new CommonException("不可选择上级组织:{}", this.getById(originDataList, sysOrg.getParentId()).getName()); } + // 更新组织 this.updateById(sysOrg); - // 发布更新事件 CommonDataChangeEventCenter.doUpdateWithData(SysDataTypeEnum.ORG.getValue(), JSONUtil.createArray().put(sysOrg)); } @@ -171,7 +173,7 @@ public class SysOrgServiceImpl extends ServiceImpl impleme List positionOrgIdList = CollectionUtil.newArrayList(); positionJsonList.forEach(positionJson -> JSONUtil.toList(JSONUtil.parseArray(positionJson), JSONObject.class) .forEach(jsonObject -> positionOrgIdList.add(jsonObject.getStr("orgId")))); - boolean hasPositionUser = CollectionUtil.intersectionDistinct(toDeleteOrgIdList, CollectionUtil.removeNull(positionOrgIdList)).size() > 0; + boolean hasPositionUser = !CollectionUtil.intersectionDistinct(toDeleteOrgIdList, CollectionUtil.removeNull(positionOrgIdList)).isEmpty(); if(hasPositionUser) { throw new CommonException("请先删除组织下的用户"); } @@ -210,43 +212,33 @@ public class SysOrgServiceImpl extends ServiceImpl impleme @Override public List getAllOrgList() { - // 从缓存中取 - Object cacheValue = commonCacheOperator.get(ORG_CACHE_ALL_KEY); - if(ObjectUtil.isNotEmpty(cacheValue)) { - return JSONUtil.toList(JSONUtil.parseArray(cacheValue), SysOrg.class); - } - List orgList = this.list(new LambdaQueryWrapper().orderByAsc(SysOrg::getSortCode)); - if(ObjectUtil.isNotEmpty(orgList)) { - // 更新到缓存 - commonCacheOperator.put(ORG_CACHE_ALL_KEY, JSONUtil.toJsonStr(orgList)); - } - return orgList; + return this.list(new LambdaQueryWrapper().orderByAsc(SysOrg::getSortCode)); } @Override public String getOrgIdByOrgFullNameWithCreate(String orgFullName) { - List cachedAllOrgList = this.getAllOrgList(); - List> treeList = TreeUtil.build(cachedAllOrgList.stream().map(sysOrg -> + List allOrgList = this.getAllOrgList(); + List> treeList = TreeUtil.build(allOrgList.stream().map(sysOrg -> new TreeNode<>(sysOrg.getId(), sysOrg.getParentId(), sysOrg.getName(), sysOrg.getSortCode())) .collect(Collectors.toList()), "0"); - return findOrgIdByOrgName("0", StrUtil.split(orgFullName, StrUtil.DASHED).iterator(), cachedAllOrgList, treeList); + return findOrgIdByOrgName("0", StrUtil.split(orgFullName, StrUtil.DASHED).iterator(), allOrgList, treeList); } - public String findOrgIdByOrgName(String parentId, Iterator iterator, List cachedAllOrgList, List> treeList) { + public String findOrgIdByOrgName(String parentId, Iterator iterator, List allOrgList, List> treeList) { String orgName = iterator.next(); if(ObjectUtil.isNotEmpty(treeList)) { List> findList = treeList.stream().filter(tree -> tree.getName().equals(orgName)).collect(Collectors.toList()); if(ObjectUtil.isNotEmpty(findList)) { if(iterator.hasNext()) { - return findOrgIdByOrgName(findList.get(0).getId(), iterator, cachedAllOrgList, findList.get(0).getChildren()); + return findOrgIdByOrgName(findList.get(0).getId(), iterator, allOrgList, findList.get(0).getChildren()); } else { return findList.get(0).getId(); } } } - String orgId = this.doCreateOrg(parentId, orgName, cachedAllOrgList); + String orgId = this.doCreateOrg(parentId, orgName, allOrgList); if(iterator.hasNext()) { - return findOrgIdByOrgName(orgId, iterator, cachedAllOrgList, CollectionUtil.newArrayList()); + return findOrgIdByOrgName(orgId, iterator, allOrgList, CollectionUtil.newArrayList()); } else { return orgId; } @@ -258,7 +250,7 @@ public class SysOrgServiceImpl extends ServiceImpl impleme * @author xuyuxiang * @date 2023/3/8 9:38 **/ - public String doCreateOrg(String parentId, String orgName, List cachedAllOrgList) { + public String doCreateOrg(String parentId, String orgName, List allOrgList) { //创建该组织 SysOrg sysOrg = new SysOrg(); sysOrg.setName(orgName); @@ -266,13 +258,12 @@ public class SysOrgServiceImpl extends ServiceImpl impleme sysOrg.setParentId(parentId); sysOrg.setCategory("0".equals(parentId)?SysOrgCategoryEnum.COMPANY.getValue():SysOrgCategoryEnum.DEPT.getValue()); sysOrg.setSortCode(99); + // 保存组织 this.save(sysOrg); + // 插入扩展信息 + sysOrgExtService.createExtInfo(sysOrg.getId(), SysOrgSourceFromTypeEnum.SYSTEM_ADD.getValue()); // 发布增加事件 CommonDataChangeEventCenter.doAddWithData(SysDataTypeEnum.ORG.getValue(), JSONUtil.createArray().put(sysOrg)); - // 将该组织加入缓存 - cachedAllOrgList.add(sysOrg); - // 更新缓存 - commonCacheOperator.put(ORG_CACHE_ALL_KEY, cachedAllOrgList); return sysOrg.getId(); } @@ -289,25 +280,27 @@ public class SysOrgServiceImpl extends ServiceImpl impleme @Override public Page orgListSelector(SysOrgSelectorOrgListParam sysOrgSelectorOrgListParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 - lambdaQueryWrapper.select(SysOrg::getId, SysOrg::getParentId, SysOrg::getName, + queryWrapper.lambda().select(SysOrg::getId, SysOrg::getParentId, SysOrg::getName, SysOrg::getCategory, SysOrg::getSortCode); if(ObjectUtil.isNotEmpty(sysOrgSelectorOrgListParam.getParentId())) { - lambdaQueryWrapper.eq(SysOrg::getParentId, sysOrgSelectorOrgListParam.getParentId()); + queryWrapper.lambda().eq(SysOrg::getParentId, sysOrgSelectorOrgListParam.getParentId()); } if(ObjectUtil.isNotEmpty(sysOrgSelectorOrgListParam.getSearchKey())) { - lambdaQueryWrapper.like(SysOrg::getName, sysOrgSelectorOrgListParam.getSearchKey()); + queryWrapper.lambda().like(SysOrg::getName, sysOrgSelectorOrgListParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysOrg::getSortCode); - return this.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysOrg::getSortCode); + return this.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } @Override public Page userSelector(SysOrgSelectorUserParam sysOrgSelectorUserParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + // 只查询状态为正常的 + queryWrapper.lambda().eq(SysUser::getUserStatus, SysUserStatusEnum.ENABLE.getValue()); // 只查询部分字段 - lambdaQueryWrapper.select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, + queryWrapper.lambda().select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, SysUser::getName, SysUser::getSortCode, SysUser::getGender, SysUser::getEntryDate); // 如果查询条件为空,则直接查询 if(ObjectUtil.isAllEmpty(sysOrgSelectorUserParam.getOrgId(), sysOrgSelectorUserParam.getSearchKey())) { @@ -318,16 +311,16 @@ public class SysOrgServiceImpl extends ServiceImpl impleme List childOrgIdList = CollStreamUtil.toList(this.getChildListById(this .getAllOrgList(), sysOrgSelectorUserParam.getOrgId(), true), SysOrg::getId); if (ObjectUtil.isNotEmpty(childOrgIdList)) { - lambdaQueryWrapper.in(SysUser::getOrgId, childOrgIdList); + queryWrapper.lambda().in(SysUser::getOrgId, childOrgIdList); } else { return new Page<>(); } } if(ObjectUtil.isNotEmpty(sysOrgSelectorUserParam.getSearchKey())) { - lambdaQueryWrapper.like(SysUser::getName, sysOrgSelectorUserParam.getSearchKey()); + queryWrapper.lambda().like(SysUser::getName, sysOrgSelectorUserParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysUser::getSortCode); - return sysUserService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysUser::getSortCode); + return sysUserService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/controller/SysPositionController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/controller/SysPositionController.java index 011a8afc..87996e55 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/controller/SysPositionController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/controller/SysPositionController.java @@ -14,10 +14,11 @@ package vip.xiaonuo.sys.modular.position.controller; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -30,6 +31,7 @@ import vip.xiaonuo.sys.modular.position.entity.SysPosition; import vip.xiaonuo.sys.modular.position.param.*; import vip.xiaonuo.sys.modular.position.service.SysPositionService; +import javax.validation.Valid; import java.util.List; /** @@ -39,6 +41,7 @@ import java.util.List; * @date 2022/4/25 20:40 */ @Tag(name = "职位控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 2) @RestController @Validated public class SysPositionController { @@ -52,6 +55,7 @@ public class SysPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取职位分页") @GetMapping("/sys/position/page") public CommonResult> page(SysPositionPageParam sysPositionPageParam) { @@ -64,6 +68,7 @@ public class SysPositionController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加职位") @CommonLog("添加职位") @PostMapping("/sys/position/add") @@ -78,6 +83,7 @@ public class SysPositionController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑职位") @CommonLog("编辑职位") @PostMapping("/sys/position/edit") @@ -92,6 +98,7 @@ public class SysPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除职位") @CommonLog("删除职位") @PostMapping("/sys/position/delete") @@ -107,6 +114,7 @@ public class SysPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取职位详情") @GetMapping("/sys/position/detail") public CommonResult detail(@Valid SysPositionIdParam sysPositionIdParam) { @@ -121,6 +129,7 @@ public class SysPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取组织树选择器") @GetMapping("/sys/position/orgTreeSelector") public CommonResult>> orgTreeSelector() { @@ -133,6 +142,7 @@ public class SysPositionController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取职位选择器") @GetMapping("/sys/position/positionSelector") public CommonResult> positionSelector(SysPositionSelectorPositionParam sysPositionSelectorPositionParam) { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/entity/SysPosition.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/entity/SysPosition.java index 1ca83e15..712336f8 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/entity/SysPosition.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/entity/SysPosition.java @@ -57,6 +57,6 @@ public class SysPosition extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/enums/SysPositionCategoryEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/enums/SysPositionCategoryEnum.java index e3d40877..78d3c6f0 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/enums/SysPositionCategoryEnum.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/enums/SysPositionCategoryEnum.java @@ -13,7 +13,6 @@ package vip.xiaonuo.sys.modular.position.enums; import lombok.Getter; -import vip.xiaonuo.common.exception.CommonException; /** * 职位分类枚举 @@ -38,11 +37,4 @@ public enum SysPositionCategoryEnum { SysPositionCategoryEnum(String value) { this.value = value; } - - public static void validate(String value) { - boolean flag = HIGH.getValue().equals(value) || MIDDLE.getValue().equals(value) || LOW.getValue().equals(value); - if(!flag) { - throw new CommonException("不支持的职位分类:{}", value); - } - } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/SysPositionMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/SysPositionMapper.java index 68b1b3cc..b186db33 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/SysPositionMapper.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/SysPositionMapper.java @@ -12,10 +12,7 @@ */ package vip.xiaonuo.sys.modular.position.mapper; -import com.baomidou.mybatisplus.annotation.InterceptorIgnore; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import org.apache.ibatis.annotations.Param; import vip.xiaonuo.sys.modular.position.entity.SysPosition; /** @@ -25,13 +22,4 @@ import vip.xiaonuo.sys.modular.position.entity.SysPosition; * @date 2022/4/21 18:37 **/ public interface SysPositionMapper extends BaseMapper { - - /** - * 删除数据并忽略插件(逻辑删除、租户拼接) - * - * @author xuyuxiang - * @date 2023/12/25 23:20 - */ - @InterceptorIgnore(tenantLine = "true") - void deleteIgnoreInterceptor(@Param("ew") LambdaQueryWrapper lambdaQueryWrapper); } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/mapping/SysPositionMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/mapping/SysPositionMapper.xml index 6b68aa65..32780b81 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/mapping/SysPositionMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/mapper/mapping/SysPositionMapper.xml @@ -2,7 +2,4 @@ - - DELETE FROM SYS_POSITION ${ew.customSqlSegment} - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionAddParam.java index 185a7e33..86ec4788 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionAddParam.java @@ -29,22 +29,22 @@ import lombok.Setter; public class SysPositionAddParam { /** 组织id */ - @Schema(description = "组织id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "组织id") @NotBlank(message = "orgId不能为空") private String orgId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionEditParam.java index 3a3901b9..237233a6 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionEditParam.java @@ -29,27 +29,27 @@ import lombok.Setter; public class SysPositionEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 组织id */ - @Schema(description = "组织id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "组织id") @NotBlank(message = "orgId不能为空") private String orgId; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionIdParam.java index 8d41825f..9b0673e3 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/param/SysPositionIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysPositionIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/provider/SysPositionApiProvider.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/provider/SysPositionApiProvider.java index d9ac37b7..f1909a53 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/provider/SysPositionApiProvider.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/provider/SysPositionApiProvider.java @@ -40,7 +40,7 @@ public class SysPositionApiProvider implements SysPositionApi { @SuppressWarnings("ALL") @Override - public Page positionSelector(String orgId, String searchKey) { + public Page positionSelector(String orgId, String searchKey, Integer current, Integer size) { SysPositionSelectorPositionParam sysPositionSelectorPositionParam = new SysPositionSelectorPositionParam(); sysPositionSelectorPositionParam.setOrgId(orgId); sysPositionSelectorPositionParam.setSearchKey(searchKey); diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/service/impl/SysPositionServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/service/impl/SysPositionServiceImpl.java index 5d73410e..2059b746 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/service/impl/SysPositionServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/position/service/impl/SysPositionServiceImpl.java @@ -91,7 +91,6 @@ public class SysPositionServiceImpl extends ServiceImpl().eq(SysPosition::getOrgId, sysPosition.getOrgId()) .eq(SysPosition::getName, sysPosition.getName())) > 0; @@ -108,7 +107,6 @@ public class SysPositionServiceImpl extends ServiceImpl().eq(SysPosition::getOrgId, sysPosition.getOrgId()) @@ -182,7 +180,7 @@ public class SysPositionServiceImpl extends ServiceImpl positionSelector(SysPositionSelectorPositionParam sysPositionSelectorPositionParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 - lambdaQueryWrapper.select(SysPosition::getId, SysPosition::getOrgId, SysPosition::getName, + queryWrapper.lambda().select(SysPosition::getId, SysPosition::getOrgId, SysPosition::getName, SysPosition::getCategory, SysPosition::getSortCode); if(ObjectUtil.isNotEmpty(sysPositionSelectorPositionParam.getOrgId())) { - lambdaQueryWrapper.eq(SysPosition::getOrgId, sysPositionSelectorPositionParam.getOrgId()); + queryWrapper.lambda().eq(SysPosition::getOrgId, sysPositionSelectorPositionParam.getOrgId()); } if(ObjectUtil.isNotEmpty(sysPositionSelectorPositionParam.getSearchKey())) { - lambdaQueryWrapper.like(SysPosition::getName, sysPositionSelectorPositionParam.getSearchKey()); + queryWrapper.lambda().like(SysPosition::getName, sysPositionSelectorPositionParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysPosition::getSortCode); - return this.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysPosition::getSortCode); + return this.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/entity/SysRelation.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/entity/SysRelation.java index 43549e09..188c1095 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/entity/SysRelation.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/entity/SysRelation.java @@ -48,6 +48,6 @@ public class SysRelation{ /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/mapper/mapping/SysRelationMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/mapper/mapping/SysRelationMapper.xml index 74148f75..5309de89 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/mapper/mapping/SysRelationMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/mapper/mapping/SysRelationMapper.xml @@ -2,5 +2,4 @@ - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/service/impl/SysRelationServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/service/impl/SysRelationServiceImpl.java index 69d77731..105eb488 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/service/impl/SysRelationServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/relation/service/impl/SysRelationServiceImpl.java @@ -72,41 +72,49 @@ public class SysRelationServiceImpl extends ServiceImpl targetIdList, String category) { this.saveRelationBatch(objectId, targetIdList, category, null, false); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithAppend(String objectId, List targetIdList, String category, List extJsonList) { this.saveRelationBatch(objectId, targetIdList, category, extJsonList, false); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationWithClear(String objectId, String targetId, String category) { this.saveRelation(objectId, targetId, category, null, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationWithClear(String objectId, String targetId, String category, String extJson) { this.saveRelation(objectId, targetId, category, extJson, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithClear(String objectId, List targetIdList, String category) { this.saveRelationBatch(objectId, targetIdList, category, null, true); } + @Transactional(rollbackFor = Exception.class) @Override public void saveRelationBatchWithClear(String objectId, List targetIdList, String category, List extJsonList) { this.saveRelationBatch(objectId, targetIdList, category, extJsonList, true); diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysButtonController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysButtonController.java index 46545a6c..b2c3971f 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysButtonController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysButtonController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.sys.modular.resource.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -32,6 +33,7 @@ import vip.xiaonuo.sys.modular.resource.param.button.SysButtonIdParam; import vip.xiaonuo.sys.modular.resource.param.button.SysButtonPageParam; import vip.xiaonuo.sys.modular.resource.service.SysButtonService; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +43,7 @@ import java.util.List; * @date 2022/6/27 13:56 **/ @Tag(name = "按钮控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 3) @RestController @Validated public class SysButtonController { @@ -54,6 +57,7 @@ public class SysButtonController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取按钮分页") @GetMapping("/sys/button/page") public CommonResult> page(SysButtonPageParam sysButtonPageParam) { @@ -66,6 +70,7 @@ public class SysButtonController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加按钮") @CommonLog("添加按钮") @PostMapping("/sys/button/add") @@ -80,6 +85,7 @@ public class SysButtonController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑按钮") @CommonLog("编辑按钮") @PostMapping("/sys/button/edit") @@ -94,6 +100,7 @@ public class SysButtonController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除按钮") @CommonLog("删除按钮") @PostMapping("/sys/button/delete") @@ -108,6 +115,7 @@ public class SysButtonController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取按钮详情") @GetMapping("/sys/button/detail") public CommonResult detail(@Valid SysButtonIdParam sysButtonIdParam) { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysMenuController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysMenuController.java index 3ccd819f..b9184d84 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysMenuController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysMenuController.java @@ -14,10 +14,11 @@ package vip.xiaonuo.sys.modular.resource.controller; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -31,6 +32,7 @@ import vip.xiaonuo.sys.modular.resource.entity.SysModule; import vip.xiaonuo.sys.modular.resource.param.menu.*; import vip.xiaonuo.sys.modular.resource.service.SysMenuService; +import javax.validation.Valid; import java.util.List; /** @@ -40,6 +42,7 @@ import java.util.List; * @date 2022/6/27 14:09 **/ @Tag(name = "菜单控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 5) @RestController @Validated public class SysMenuController { @@ -53,6 +56,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取菜单分页") @GetMapping("/sys/menu/page") public CommonResult> page(SysMenuPageParam sysMenuPageParam) { @@ -65,6 +69,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 2) @Operation(summary = "获取菜单树") @GetMapping("/sys/menu/tree") public CommonResult>> tree(SysMenuTreeParam sysMenuTreeParam) { @@ -77,6 +82,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "添加菜单") @CommonLog("添加菜单") @PostMapping("/sys/menu/add") @@ -91,6 +97,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 4) @Operation(summary = "编辑菜单") @CommonLog("编辑菜单") @PostMapping("/sys/menu/edit") @@ -105,6 +112,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 5) @Operation(summary = "更改菜单所属模块") @CommonLog("更改菜单所属模块") @PostMapping("/sys/menu/changeModule") @@ -119,6 +127,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "删除菜单") @CommonLog("删除菜单") @PostMapping("/sys/menu/delete") @@ -134,6 +143,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "获取菜单详情") @GetMapping("/sys/menu/detail") public CommonResult detail(@Valid SysMenuIdParam sysMenuIdParam) { @@ -148,6 +158,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 8) @Operation(summary = "获取模块选择器") @GetMapping("/sys/menu/moduleSelector") public CommonResult> moduleSelector(SysMenuSelectorModuleParam sysMenuSelectorModuleParam) { @@ -160,6 +171,7 @@ public class SysMenuController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 9) @Operation(summary = "获取菜单树选择器") @GetMapping("/sys/menu/menuTreeSelector") public CommonResult>> menuTreeSelector(SysMenuSelectorMenuParam sysMenuSelectorMenuParam) { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysModuleController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysModuleController.java index d1205b7a..108165d4 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysModuleController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/controller/SysModuleController.java @@ -13,10 +13,11 @@ package vip.xiaonuo.sys.modular.resource.controller; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -32,6 +33,7 @@ import vip.xiaonuo.sys.modular.resource.param.module.SysModuleIdParam; import vip.xiaonuo.sys.modular.resource.param.module.SysModulePageParam; import vip.xiaonuo.sys.modular.resource.service.SysModuleService; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +43,7 @@ import java.util.List; * @date 2022/6/27 14:12 **/ @Tag(name = "模块控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 6) @RestController @Validated public class SysModuleController { @@ -54,6 +57,7 @@ public class SysModuleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取模块分页") @GetMapping("/sys/module/page") public CommonResult> page(SysModulePageParam sysModulePageParam) { @@ -66,6 +70,7 @@ public class SysModuleController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加模块") @CommonLog("添加模块") @PostMapping("/sys/module/add") @@ -80,6 +85,7 @@ public class SysModuleController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑模块") @CommonLog("编辑模块") @PostMapping("/sys/module/edit") @@ -94,11 +100,12 @@ public class SysModuleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除模块") @CommonLog("删除模块") @PostMapping("/sys/module/delete") public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List sysModuleIdParamList) { + List sysModuleIdParamList) { sysModuleService.delete(sysModuleIdParamList); return CommonResult.ok(); } @@ -109,6 +116,7 @@ public class SysModuleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取模块详情") @GetMapping("/sys/module/detail") public CommonResult detail(@Valid SysModuleIdParam sysModuleIdParam) { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysButton.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysButton.java index 62d57a5c..c64de15b 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysButton.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysButton.java @@ -57,6 +57,6 @@ public class SysButton extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysMenu.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysMenu.java index a229bb66..bccb9c75 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysMenu.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysMenu.java @@ -45,7 +45,7 @@ public class SysMenu extends CommonEntity { /** 别名 */ @Schema(description = "别名") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String name; /** 编码 */ @@ -70,11 +70,11 @@ public class SysMenu extends CommonEntity { /** 组件 */ @Schema(description = "组件") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String component; /** 图标 */ - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Schema(description = "图标") private String icon; @@ -92,6 +92,6 @@ public class SysMenu extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysModule.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysModule.java index f3009018..6730e815 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysModule.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/entity/SysModule.java @@ -61,6 +61,6 @@ public class SysModule extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysButtonMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysButtonMapper.java index a30c6bff..24ed63d1 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysButtonMapper.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysButtonMapper.java @@ -12,10 +12,7 @@ */ package vip.xiaonuo.sys.modular.resource.mapper; -import com.baomidou.mybatisplus.annotation.InterceptorIgnore; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import org.apache.ibatis.annotations.Param; import vip.xiaonuo.sys.modular.resource.entity.SysButton; /** @@ -25,13 +22,4 @@ import vip.xiaonuo.sys.modular.resource.entity.SysButton; * @date 2022/4/21 18:37 **/ public interface SysButtonMapper extends BaseMapper { - - /** - * 删除数据并忽略插件(逻辑删除、租户拼接) - * - * @author xuyuxiang - * @date 2023/12/25 23:20 - */ - @InterceptorIgnore(tenantLine = "true") - void deleteIgnoreInterceptor(@Param("ew") LambdaQueryWrapper lambdaQueryWrapper); } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysMenuMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysMenuMapper.java index c334b404..0191d82c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysMenuMapper.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysMenuMapper.java @@ -12,10 +12,7 @@ */ package vip.xiaonuo.sys.modular.resource.mapper; -import com.baomidou.mybatisplus.annotation.InterceptorIgnore; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import org.apache.ibatis.annotations.Param; import vip.xiaonuo.sys.modular.resource.entity.SysMenu; /** @@ -25,13 +22,4 @@ import vip.xiaonuo.sys.modular.resource.entity.SysMenu; * @date 2022/4/21 18:37 **/ public interface SysMenuMapper extends BaseMapper { - - /** - * 删除数据并忽略插件(逻辑删除、租户拼接) - * - * @author xuyuxiang - * @date 2023/12/25 23:20 - */ - @InterceptorIgnore(tenantLine = "true") - void deleteIgnoreInterceptor(@Param("ew") LambdaQueryWrapper lambdaQueryWrapper); } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysModuleMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysModuleMapper.java index 0c398baf..e5ad36cd 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysModuleMapper.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/SysModuleMapper.java @@ -12,10 +12,7 @@ */ package vip.xiaonuo.sys.modular.resource.mapper; -import com.baomidou.mybatisplus.annotation.InterceptorIgnore; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import org.apache.ibatis.annotations.Param; import vip.xiaonuo.sys.modular.resource.entity.SysModule; /** @@ -25,14 +22,4 @@ import vip.xiaonuo.sys.modular.resource.entity.SysModule; * @date 2022/4/21 18:37 **/ public interface SysModuleMapper extends BaseMapper { - - /** - * 删除数据并忽略插件(逻辑删除、租户拼接) - * - * @author xuyuxiang - * @date 2023/12/25 23:20 - */ - @InterceptorIgnore(tenantLine = "true") - void deleteIgnoreInterceptor(@Param("ew") LambdaQueryWrapper lambdaQueryWrapper); - } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysButtonMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysButtonMapper.xml index a0864ada..14094f59 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysButtonMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysButtonMapper.xml @@ -2,7 +2,4 @@ - - DELETE FROM SYS_RESOURCE ${ew.customSqlSegment} - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysMenuMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysMenuMapper.xml index 56ebe153..afab2463 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysMenuMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysMenuMapper.xml @@ -2,7 +2,4 @@ - - DELETE FROM SYS_RESOURCE ${ew.customSqlSegment} - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysModuleMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysModuleMapper.xml index 06e5d789..1666db84 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysModuleMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/mapper/mapping/SysModuleMapper.xml @@ -2,7 +2,4 @@ - - DELETE FROM SYS_RESOURCE ${ew.customSqlSegment} - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonAddParam.java index af9be7d3..71891b66 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonAddParam.java @@ -29,22 +29,22 @@ import lombok.Setter; public class SysButtonAddParam { /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 标题 */ - @Schema(description = "标题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "标题") @NotBlank(message = "title不能为空") private String title; /** 编码 */ - @Schema(description = "编码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "编码") @NotBlank(message = "code不能为空") private String code; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonEditParam.java index 7cde31d4..05838ed9 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonEditParam.java @@ -29,27 +29,27 @@ import lombok.Setter; public class SysButtonEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 标题 */ - @Schema(description = "标题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "标题") @NotBlank(message = "title不能为空") private String title; /** 编码 */ - @Schema(description = "编码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "编码") @NotBlank(message = "code不能为空") private String code; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonIdParam.java index d7d1092f..b10de451 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/button/SysButtonIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysButtonIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuAddParam.java index 178938ba..31d9a8ca 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuAddParam.java @@ -29,32 +29,32 @@ import lombok.Setter; public class SysMenuAddParam { /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 标题 */ - @Schema(description = "标题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "标题") @NotBlank(message = "title不能为空") private String title; /** 菜单类型 */ - @Schema(description = "菜单类型", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "菜单类型") @NotBlank(message = "menuType不能为空") private String menuType; /** 模块 */ - @Schema(description = "模块", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "模块") @NotBlank(message = "module不能为空") private String module; /** 路径 */ - @Schema(description = "路径", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "路径") @NotBlank(message = "path不能为空") private String path; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuChangeModuleParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuChangeModuleParam.java index faf98946..488dea39 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuChangeModuleParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuChangeModuleParam.java @@ -28,12 +28,12 @@ import lombok.Setter; public class SysMenuChangeModuleParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 模块 */ - @Schema(description = "module", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "module") @NotBlank(message = "module不能为空") private String module; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuEditParam.java index 87b8c3e4..60256834 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuEditParam.java @@ -29,37 +29,37 @@ import lombok.Setter; public class SysMenuEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 父id */ - @Schema(description = "父id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "父id") @NotBlank(message = "parentId不能为空") private String parentId; /** 标题 */ - @Schema(description = "标题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "标题") @NotBlank(message = "title不能为空") private String title; /** 菜单类型 */ - @Schema(description = "菜单类型", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "菜单类型") @NotBlank(message = "menuType不能为空") private String menuType; /** 模块 */ - @Schema(description = "模块", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "模块") @NotBlank(message = "module不能为空") private String module; /** 路径 */ - @Schema(description = "路径", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "路径") @NotBlank(message = "path不能为空") private String path; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuIdParam.java index 21870abd..02a49bcb 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/menu/SysMenuIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysMenuIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleAddParam.java index 8b2992e0..e202c884 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleAddParam.java @@ -29,22 +29,22 @@ import lombok.Setter; public class SysModuleAddParam { /** 标题 */ - @Schema(description = "标题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "标题") @NotBlank(message = "title不能为空") private String title; /** 图标 */ - @Schema(description = "图标", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "图标") @NotBlank(message = "icon不能为空") private String icon; /** 颜色 */ - @Schema(description = "颜色", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "颜色") @NotBlank(message = "color不能为空") private String color; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleEditParam.java index c401151b..6b5e27f1 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleEditParam.java @@ -29,27 +29,27 @@ import lombok.Setter; public class SysModuleEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 标题 */ - @Schema(description = "标题", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "标题") @NotBlank(message = "title不能为空") private String title; /** 图标 */ - @Schema(description = "图标", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "图标") @NotBlank(message = "icon不能为空") private String icon; /** 颜色 */ - @Schema(description = "颜色", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "颜色") @NotBlank(message = "color不能为空") private String color; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleIdParam.java index 46bfa2fb..e4f44639 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/param/module/SysModuleIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysModuleIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysButtonServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysButtonServiceImpl.java index dbcd9251..58567add 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysButtonServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysButtonServiceImpl.java @@ -16,6 +16,7 @@ import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.collection.CollStreamUtil; import cn.hutool.core.collection.CollectionUtil; import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.RandomUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; @@ -96,6 +97,7 @@ public class SysButtonServiceImpl extends ServiceImpl sysButtonIdParamList) { List buttonIdList = CollStreamUtil.toList(sysButtonIdParamList, SysButtonIdParam::getId); diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysMenuServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysMenuServiceImpl.java index 1a0f9bc7..35395609 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysMenuServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysMenuServiceImpl.java @@ -92,16 +92,16 @@ public class SysMenuServiceImpl extends ServiceImpl impl @Override public List> tree(SysMenuTreeParam sysMenuTreeParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); - lambdaQueryWrapper.eq(SysMenu::getCategory, SysResourceCategoryEnum.MENU.getValue()) + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + queryWrapper.lambda().eq(SysMenu::getCategory, SysResourceCategoryEnum.MENU.getValue()) .orderByAsc(SysMenu::getSortCode); if(ObjectUtil.isNotEmpty(sysMenuTreeParam.getModule())) { - lambdaQueryWrapper.eq(SysMenu::getModule, sysMenuTreeParam.getModule()); + queryWrapper.lambda().eq(SysMenu::getModule, sysMenuTreeParam.getModule()); } if(ObjectUtil.isNotEmpty(sysMenuTreeParam.getSearchKey())) { - lambdaQueryWrapper.like(SysMenu::getTitle, sysMenuTreeParam.getSearchKey()); + queryWrapper.lambda().like(SysMenu::getTitle, sysMenuTreeParam.getSearchKey()); } - List resourceList = this.list(lambdaQueryWrapper); + List resourceList = this.list(queryWrapper.lambda()); // 填充上层的父级菜单 this.fillParentSysMenuInfo(resourceList); @@ -165,6 +165,7 @@ public class SysMenuServiceImpl extends ServiceImpl impl throw new CommonException("module与上级菜单不一致"); } } + sysMenu.setCode(RandomUtil.randomString(10)); sysMenu.setCategory(SysResourceCategoryEnum.MENU.getValue()); this.save(sysMenu); @@ -273,6 +274,7 @@ public class SysMenuServiceImpl extends ServiceImpl impl CommonDataChangeEventCenter.doUpdateWithData(SysDataTypeEnum.RESOURCE.getValue(), JSONUtil.createArray().put(sysMenu)); } + @Transactional(rollbackFor = Exception.class) @Override public void changeModule(SysMenuChangeModuleParam sysMenuChangeModuleParam) { SysMenu sysMenu = this.queryEntity(sysMenuChangeModuleParam.getId()); @@ -319,6 +321,7 @@ public class SysMenuServiceImpl extends ServiceImpl impl } } + @Transactional(rollbackFor = Exception.class) @Override public void delete(List sysMenuIdParamList) { List sysMenuIdList = CollStreamUtil.toList(sysMenuIdParamList, SysMenuIdParam::getId); @@ -359,12 +362,12 @@ public class SysMenuServiceImpl extends ServiceImpl impl @Override public List moduleSelector(SysMenuSelectorModuleParam sysMenuSelectorModuleParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); if(ObjectUtil.isNotEmpty(sysMenuSelectorModuleParam.getSearchKey())) { - lambdaQueryWrapper.like(SysModule::getTitle, sysMenuSelectorModuleParam.getSearchKey()); + queryWrapper.lambda().like(SysModule::getTitle, sysMenuSelectorModuleParam.getSearchKey()); } - lambdaQueryWrapper.eq(SysModule::getCategory, SysResourceCategoryEnum.MODULE.getValue()); - return sysModuleService.list(lambdaQueryWrapper); + queryWrapper.lambda().eq(SysModule::getCategory, SysResourceCategoryEnum.MODULE.getValue()); + return sysModuleService.list(queryWrapper.lambda()); } @Override diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysModuleServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysModuleServiceImpl.java index 732483d4..ad6b367c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysModuleServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/resource/service/impl/SysModuleServiceImpl.java @@ -95,7 +95,6 @@ public class SysModuleServiceImpl extends ServiceImpl sysModuleIdParamList) { List sysModuleIdList = CollStreamUtil.toList(sysModuleIdParamList, SysModuleIdParam::getId); diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/controller/SysRoleController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/controller/SysRoleController.java index 60919b6c..5913c23e 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/controller/SysRoleController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/controller/SysRoleController.java @@ -14,10 +14,11 @@ package vip.xiaonuo.sys.modular.role.controller; import cn.hutool.core.lang.tree.Tree; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.GetMapping; @@ -32,6 +33,7 @@ import vip.xiaonuo.sys.modular.role.result.*; import vip.xiaonuo.sys.modular.role.service.SysRoleService; import vip.xiaonuo.sys.modular.user.entity.SysUser; +import javax.validation.Valid; import java.util.List; /** @@ -41,6 +43,7 @@ import java.util.List; * @date 2022/4/25 20:19 */ @Tag(name = "角色控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 8) @RestController @Validated public class SysRoleController { @@ -54,6 +57,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取角色分页") @GetMapping("/sys/role/page") public CommonResult> page(SysRolePageParam sysRolePageParam) { @@ -66,6 +70,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加角色") @CommonLog("添加角色") @PostMapping("/sys/role/add") @@ -80,6 +85,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑角色") @CommonLog("编辑角色") @PostMapping("/sys/role/edit") @@ -94,6 +100,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除角色") @CommonLog("删除角色") @PostMapping("/sys/role/delete") @@ -109,6 +116,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取角色详情") @GetMapping("/sys/role/detail") public CommonResult detail(@Valid SysRoleIdParam sysRoleIdParam) { @@ -121,6 +129,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 6) @Operation(summary = "获取角色拥有资源") @GetMapping("/sys/role/ownResource") public CommonResult ownResource(@Valid SysRoleIdParam sysRoleIdParam) { @@ -133,6 +142,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 7) @Operation(summary = "给角色授权资源") @CommonLog("给角色授权资源") @PostMapping("/sys/role/grantResource") @@ -147,6 +157,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 8) @Operation(summary = "获取角色拥有移动端菜单") @GetMapping("/sys/role/ownMobileMenu") public CommonResult ownMobileMenu(@Valid SysRoleIdParam sysRoleIdParam) { @@ -159,6 +170,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 9) @Operation(summary = "给角色授权移动端菜单") @CommonLog("给角色授权移动端菜单") @PostMapping("/sys/role/grantMobileMenu") @@ -173,6 +185,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 10) @Operation(summary = "获取角色拥有权限") @GetMapping("/sys/role/ownPermission") public CommonResult ownPermission(@Valid SysRoleIdParam sysRoleIdParam) { @@ -185,6 +198,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 11) @Operation(summary = "给角色授权权限") @CommonLog("给角色授权权限") @PostMapping("/sys/role/grantPermission") @@ -199,6 +213,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 12) @Operation(summary = "获取角色下的用户") @GetMapping("/sys/role/ownUser") public CommonResult> ownUser(@Valid SysRoleIdParam sysRoleIdParam) { @@ -211,6 +226,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 13) @Operation(summary = "给角色授权用户") @CommonLog("给角色授权用户") @PostMapping("/sys/role/grantUser") @@ -227,6 +243,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 14) @Operation(summary = "获取组织树选择器") @GetMapping("/sys/role/orgTreeSelector") public CommonResult>> orgTreeSelector() { @@ -239,10 +256,11 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 15) @Operation(summary = "获取资源授权树") @GetMapping("/sys/role/resourceTreeSelector") public CommonResult> resourceTreeSelector() { - return CommonResult.data(sysRoleService.resourceTreeSelector()); + return CommonResult.data(sysRoleService.resourceTreeSelector(true)); } /** @@ -251,6 +269,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 16) @Operation(summary = "获取移动端菜单授权树") @GetMapping("/sys/role/mobileMenuTreeSelector") public CommonResult> mobileMenuTreeSelector() { @@ -263,6 +282,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 17) @Operation(summary = "获取权限授权树") @GetMapping("/sys/role/permissionTreeSelector") public CommonResult> permissionTreeSelector() { @@ -275,6 +295,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 18) @Operation(summary = "获取角色选择器") @GetMapping("/sys/role/roleSelector") public CommonResult> roleSelector(SysRoleSelectorRoleParam sysRoleSelectorRoleParam) { @@ -287,6 +308,7 @@ public class SysRoleController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 19) @Operation(summary = "获取用户选择器") @GetMapping("/sys/role/userSelector") public CommonResult> userSelector(SysRoleSelectorUserParam sysRoleSelectorUserParam) { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/entity/SysRole.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/entity/SysRole.java index b394b557..8a60c5a0 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/entity/SysRole.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/entity/SysRole.java @@ -37,7 +37,7 @@ public class SysRole extends CommonEntity { /** 组织id */ @Schema(description = "组织id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String orgId; /** 名称 */ @@ -58,6 +58,6 @@ public class SysRole extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/SysRoleMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/SysRoleMapper.java index c79b6237..b98ffb40 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/SysRoleMapper.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/SysRoleMapper.java @@ -12,10 +12,7 @@ */ package vip.xiaonuo.sys.modular.role.mapper; -import com.baomidou.mybatisplus.annotation.InterceptorIgnore; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import org.apache.ibatis.annotations.Param; import vip.xiaonuo.sys.modular.role.entity.SysRole; /** @@ -25,13 +22,4 @@ import vip.xiaonuo.sys.modular.role.entity.SysRole; * @date 2022/4/21 18:37 **/ public interface SysRoleMapper extends BaseMapper { - - /** - * 删除数据并忽略插件(逻辑删除、租户拼接) - * - * @author xuyuxiang - * @date 2023/12/25 23:20 - */ - @InterceptorIgnore(tenantLine = "true") - void deleteIgnoreInterceptor(@Param("ew") LambdaQueryWrapper lambdaQueryWrapper); } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/mapping/SysRoleMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/mapping/SysRoleMapper.xml index de05f4c5..e0b69f04 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/mapping/SysRoleMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/mapper/mapping/SysRoleMapper.xml @@ -2,7 +2,4 @@ - - DELETE FROM SYS_ROLE ${ew.customSqlSegment} - \ No newline at end of file diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleAddParam.java index e126a3e7..ed96e2c8 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleAddParam.java @@ -29,17 +29,22 @@ import lombok.Setter; public class SysRoleAddParam { /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; + /** 编码 */ + @Schema(description = "编码") + @NotBlank(message = "code不能为空") + private String code; + /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleEditParam.java index f69b1c67..a17b3a81 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleEditParam.java @@ -29,22 +29,27 @@ import lombok.Setter; public class SysRoleEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 名称 */ - @Schema(description = "名称", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "名称") @NotBlank(message = "name不能为空") private String name; + /** 编码 */ + @Schema(description = "编码") + @NotBlank(message = "code不能为空") + private String code; + /** 分类 */ - @Schema(description = "分类", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "分类") @NotBlank(message = "category不能为空") private String category; /** 排序码 */ - @Schema(description = "排序码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "排序码") @NotNull(message = "sortCode不能为空") private Integer sortCode; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantMobileMenuParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantMobileMenuParam.java index 61abec87..936257c5 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantMobileMenuParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantMobileMenuParam.java @@ -13,12 +13,12 @@ package vip.xiaonuo.sys.modular.role.param; import io.swagger.v3.oas.annotations.media.Schema; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; import lombok.Getter; import lombok.Setter; +import javax.validation.Valid; import java.util.List; /** @@ -32,13 +32,13 @@ import java.util.List; public class SysRoleGrantMobileMenuParam { /** 角色id */ - @Schema(description = "角色id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "角色id") @NotBlank(message = "id不能为空") private String id; /** 授权移动端菜单信息 */ @Valid - @Schema(description = "授权移动端菜单信息", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "授权移动端菜单信息") @NotNull(message = "grantInfoList不能为空") private List grantInfoList; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantPermissionParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantPermissionParam.java index d43764d4..c711cd3b 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantPermissionParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantPermissionParam.java @@ -13,12 +13,12 @@ package vip.xiaonuo.sys.modular.role.param; import io.swagger.v3.oas.annotations.media.Schema; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; import lombok.Getter; import lombok.Setter; +import javax.validation.Valid; import java.util.List; /** @@ -32,13 +32,13 @@ import java.util.List; public class SysRoleGrantPermissionParam { /** 角色id */ - @Schema(description = "角色id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "角色id") @NotBlank(message = "id不能为空") private String id; /** 授权权限信息 */ @Valid - @Schema(description = "授权权限信息", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "授权权限信息") @NotNull(message = "grantInfoList不能为空") private List grantInfoList; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantResourceParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantResourceParam.java index 64a5832d..2b0ffc42 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantResourceParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantResourceParam.java @@ -13,12 +13,12 @@ package vip.xiaonuo.sys.modular.role.param; import io.swagger.v3.oas.annotations.media.Schema; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; import lombok.Getter; import lombok.Setter; +import javax.validation.Valid; import java.util.List; /** @@ -32,13 +32,13 @@ import java.util.List; public class SysRoleGrantResourceParam { /** 角色id */ - @Schema(description = "角色id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "角色id") @NotBlank(message = "id不能为空") private String id; /** 授权资源信息 */ @Valid - @Schema(description = "授权资源信息", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "授权资源信息") @NotNull(message = "grantInfoList不能为空") private List grantInfoList; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantUserParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantUserParam.java index c3c89058..385823cf 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantUserParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleGrantUserParam.java @@ -31,12 +31,12 @@ import java.util.List; public class SysRoleGrantUserParam { /** 角色id */ - @Schema(description = "角色id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "角色id") @NotBlank(message = "id不能为空") private String id; /** 授权用户信息 */ - @Schema(description = "授权用户信息", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "授权用户信息") @NotNull(message = "grantInfoList不能为空") private List grantInfoList; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleIdParam.java index 80fb59ce..d8241507 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/param/SysRoleIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysRoleIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/provider/SysRoleApiProvider.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/provider/SysRoleApiProvider.java index c9907375..026f9b38 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/provider/SysRoleApiProvider.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/provider/SysRoleApiProvider.java @@ -107,4 +107,14 @@ public class SysRoleApiProvider implements SysRoleApi { } sysRelationService.saveRelationBatchWithAppend(superAdminRoleId, menuIdList, SysRelationCategoryEnum.SYS_ROLE_HAS_RESOURCE.getValue(), extJsonList); } + + @Override + public List resourceTreeSelector() { + return sysRoleService.resourceTreeSelector(false).stream().map(JSONUtil::parseObj).collect(Collectors.toList()); + } + + @Override + public List permissionTreeSelector() { + return sysRoleService.permissionTreeSelector(); + } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/result/SysRoleGrantResourceTreeResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/result/SysRoleGrantResourceTreeResult.java index 2fa1c6c7..6be2fa5c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/result/SysRoleGrantResourceTreeResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/result/SysRoleGrantResourceTreeResult.java @@ -28,8 +28,8 @@ import java.util.List; @Setter public class SysRoleGrantResourceTreeResult { - /** 模块id */ - @Schema(description = "模块id") + /** 模块主键 */ + @Schema(description = "模块主键") private String id; /** 模块名称*/ @@ -54,12 +54,12 @@ public class SysRoleGrantResourceTreeResult { @Setter public static class SysRoleGrantResourceMenuResult { - /** 菜单id */ - @Schema(description = "菜单id") + /** id */ + @Schema(description = "菜单主键") private String id; /** 父id */ - @Schema(description = "父id") + @Schema(description = "菜单id") private String parentId; /** 父名称 */ @@ -88,8 +88,8 @@ public class SysRoleGrantResourceTreeResult { @Setter public static class SysRoleGrantResourceButtonResult { - /** 按钮id */ - @Schema(description = "按钮id") + /** id */ + @Schema(description = "按钮主键") private String id; /** 标题 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/SysRoleService.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/SysRoleService.java index 71286ed3..36b5d416 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/SysRoleService.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/SysRoleService.java @@ -13,8 +13,10 @@ package vip.xiaonuo.sys.modular.role.service; import cn.hutool.core.lang.tree.Tree; +import cn.hutool.json.JSONObject; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.sys.modular.resource.entity.SysMenu; import vip.xiaonuo.sys.modular.role.entity.SysRole; import vip.xiaonuo.sys.modular.role.param.*; import vip.xiaonuo.sys.modular.role.result.*; @@ -158,7 +160,15 @@ public interface SysRoleService extends IService { * @author xuyuxiang * @date 2022/4/24 20:08 */ - List resourceTreeSelector(); + List resourceTreeSelector(boolean containsTen); + + /** + * 获取资源授权树 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + List resourceTreeSelector(List originDataList); /** * 获取移动端菜单授权树 @@ -168,6 +178,14 @@ public interface SysRoleService extends IService { */ List mobileMenuTreeSelector(); + /** + * 获取移动端菜单授权树 + * + * @author xuyuxiang + * @date 2022/4/24 20:08 + */ + List mobileMenuTreeSelector(List originDataList); + /** * 获取权限授权树 * diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/impl/SysRoleServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/impl/SysRoleServiceImpl.java index 1dcbfe74..a73be5a2 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/impl/SysRoleServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/role/service/impl/SysRoleServiceImpl.java @@ -20,7 +20,6 @@ import cn.hutool.core.lang.tree.Tree; import cn.hutool.core.lang.tree.TreeNode; import cn.hutool.core.lang.tree.TreeUtil; import cn.hutool.core.util.ObjectUtil; -import cn.hutool.core.util.RandomUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.extra.spring.SpringUtil; import cn.hutool.json.JSONObject; @@ -61,6 +60,7 @@ import vip.xiaonuo.sys.modular.role.param.*; import vip.xiaonuo.sys.modular.role.result.*; import vip.xiaonuo.sys.modular.role.service.SysRoleService; import vip.xiaonuo.sys.modular.user.entity.SysUser; +import vip.xiaonuo.sys.modular.user.enums.SysUserStatusEnum; import vip.xiaonuo.sys.modular.user.service.SysUserService; import java.util.ArrayList; @@ -100,7 +100,7 @@ public class SysRoleServiceImpl extends ServiceImpl impl public Page page(SysRolePageParam sysRolePageParam) { QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 - queryWrapper.lambda().select(SysRole::getId, SysRole::getOrgId, SysRole::getName, + queryWrapper.lambda().select(SysRole::getId, SysRole::getOrgId, SysRole::getName, SysRole::getCode, SysRole::getCategory, SysRole::getSortCode); if(ObjectUtil.isNotEmpty(sysRolePageParam.getOrgId())) { queryWrapper.lambda().eq(SysRole::getOrgId, sysRolePageParam.getOrgId()); @@ -124,6 +124,15 @@ public class SysRoleServiceImpl extends ServiceImpl impl @Transactional(rollbackFor = Exception.class) @Override public void add(SysRoleAddParam sysRoleAddParam) { + checkParam(sysRoleAddParam); + SysRole sysRole = BeanUtil.toBean(sysRoleAddParam, SysRole.class); + this.save(sysRole); + + // 发布增加事件 + CommonDataChangeEventCenter.doAddWithData(SysDataTypeEnum.ROLE.getValue(), JSONUtil.createArray().put(sysRole)); + } + + private void checkParam(SysRoleAddParam sysRoleAddParam) { SysRoleCategoryEnum.validate(sysRoleAddParam.getCategory()); if(SysRoleCategoryEnum.ORG.getValue().equals(sysRoleAddParam.getCategory())) { if(ObjectUtil.isEmpty(sysRoleAddParam.getOrgId())) { @@ -132,47 +141,35 @@ public class SysRoleServiceImpl extends ServiceImpl impl } else { sysRoleAddParam.setOrgId(null); } - SysRole sysRole = BeanUtil.toBean(sysRoleAddParam, SysRole.class); - boolean repeatName = this.count(new LambdaQueryWrapper().eq(SysRole::getOrgId, sysRole.getOrgId()) - .eq(SysRole::getName, sysRole.getName())) > 0; + boolean repeatName = this.count(new LambdaQueryWrapper().eq(SysRole::getOrgId, sysRoleAddParam.getOrgId()) + .eq(SysRole::getName, sysRoleAddParam.getName())) > 0; if(repeatName) { - if(ObjectUtil.isEmpty(sysRole.getOrgId())) { - throw new CommonException("存在重复的全局角色,名称为:{}", sysRole.getName()); + if(ObjectUtil.isEmpty(sysRoleAddParam.getOrgId())) { + throw new CommonException("存在重复的全局角色,名称为:{}", sysRoleAddParam.getName()); } else { - throw new CommonException("同组织下存在重复的角色,名称为:{}", sysRole.getName()); + throw new CommonException("同组织下存在重复的角色,名称为:{}", sysRoleAddParam.getName()); + } + } + boolean repeatCode = this.count(new LambdaQueryWrapper().eq(SysRole::getOrgId, sysRoleAddParam.getOrgId()) + .eq(SysRole::getCode, sysRoleAddParam.getCode())) > 0; + if(repeatCode) { + if(ObjectUtil.isEmpty(sysRoleAddParam.getOrgId())) { + throw new CommonException("存在重复的全局角色,编码为:{}", sysRoleAddParam.getCode()); + } else { + throw new CommonException("同组织下存在重复的角色,编码为:{}", sysRoleAddParam.getCode()); } } - sysRole.setCode(RandomUtil.randomString(10)); - this.save(sysRole); - - // 发布增加事件 - CommonDataChangeEventCenter.doAddWithData(SysDataTypeEnum.ROLE.getValue(), JSONUtil.createArray().put(sysRole)); } @Transactional(rollbackFor = Exception.class) @Override public void edit(SysRoleEditParam sysRoleEditParam) { SysRole sysRole = this.queryEntity(sysRoleEditParam.getId()); + checkParam(sysRoleEditParam); boolean superRole = sysRole.getCode().equals(SysBuildInEnum.BUILD_IN_ROLE_CODE.getValue()); if(superRole) { throw new CommonException("不可编辑超管角色"); } - if(SysRoleCategoryEnum.ORG.getValue().equals(sysRoleEditParam.getCategory())) { - if (ObjectUtil.isEmpty(sysRoleEditParam.getOrgId())) { - throw new CommonException("orgId不能为空"); - } - } else { - sysRoleEditParam.setOrgId(null); - } - boolean repeatName = this.count(new LambdaQueryWrapper().eq(SysRole::getOrgId, sysRole.getOrgId()) - .eq(SysRole::getName, sysRole.getName()).ne(SysRole::getId, sysRole.getId())) > 0; - if(repeatName) { - if(ObjectUtil.isEmpty(sysRole.getOrgId())) { - throw new CommonException("存在重复的全局角色,名称为:{}", sysRole.getName()); - } else { - throw new CommonException("同组织下存在重复的角色,名称为:{}", sysRole.getName()); - } - } BeanUtil.copyProperties(sysRoleEditParam, sysRole); this.updateById(sysRole); @@ -180,6 +177,36 @@ public class SysRoleServiceImpl extends ServiceImpl impl CommonDataChangeEventCenter.doUpdateWithData(SysDataTypeEnum.ROLE.getValue(), JSONUtil.createArray().put(sysRole)); } + private void checkParam(SysRoleEditParam sysRoleEditParam) { + SysRoleCategoryEnum.validate(sysRoleEditParam.getCategory()); + if(SysRoleCategoryEnum.ORG.getValue().equals(sysRoleEditParam.getCategory())) { + if(ObjectUtil.isEmpty(sysRoleEditParam.getOrgId())) { + throw new CommonException("orgId不能为空"); + } + } else { + sysRoleEditParam.setOrgId(null); + } + boolean repeatName = this.count(new LambdaQueryWrapper().eq(SysRole::getOrgId, sysRoleEditParam.getOrgId()) + .eq(SysRole::getName, sysRoleEditParam.getName()).ne(SysRole::getId, sysRoleEditParam.getId())) > 0; + if(repeatName) { + if(ObjectUtil.isEmpty(sysRoleEditParam.getOrgId())) { + throw new CommonException("存在重复的全局角色,名称为:{}", sysRoleEditParam.getName()); + } else { + throw new CommonException("同组织下存在重复的角色,名称为:{}", sysRoleEditParam.getName()); + } + } + boolean repeatCode = this.count(new LambdaQueryWrapper().eq(SysRole::getOrgId, sysRoleEditParam.getOrgId()) + .eq(SysRole::getCode, sysRoleEditParam.getCode()).ne(SysRole::getId, sysRoleEditParam.getId())) > 0; + if(repeatCode) { + if(ObjectUtil.isEmpty(sysRoleEditParam.getOrgId())) { + throw new CommonException("存在重复的全局角色,编码为:{}", sysRoleEditParam.getCode()); + } else { + throw new CommonException("同组织下存在重复的角色,编码为:{}", sysRoleEditParam.getCode()); + } + } + } + + @Transactional(rollbackFor = Exception.class) @Override public void delete(List sysRoleIdParamList) { @@ -196,6 +223,9 @@ public class SysRoleServiceImpl extends ServiceImpl impl // 级联删除角色与资源关系 sysRelationService.remove(new LambdaUpdateWrapper().in(SysRelation::getObjectId, sysRoleIdList) .eq(SysRelation::getCategory, SysRelationCategoryEnum.SYS_ROLE_HAS_RESOURCE.getValue())); + // 级联删除角色与移动端资源关系 + sysRelationService.remove(new LambdaUpdateWrapper().in(SysRelation::getObjectId, sysRoleIdList) + .eq(SysRelation::getCategory, SysRelationCategoryEnum.SYS_ROLE_HAS_MOBILE_MENU.getValue())); // 级联删除角色与权限关系 sysRelationService.remove(new LambdaUpdateWrapper().in(SysRelation::getObjectId, sysRoleIdList) .eq(SysRelation::getCategory, SysRelationCategoryEnum.SYS_ROLE_HAS_PERMISSION.getValue())); @@ -328,15 +358,22 @@ public class SysRoleServiceImpl extends ServiceImpl impl } @Override - public List resourceTreeSelector() { + public List resourceTreeSelector(boolean containsTen) { LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); lambdaQueryWrapper.in(SysMenu::getCategory, SysResourceCategoryEnum.MODULE.getValue(), SysResourceCategoryEnum.MENU.getValue(), SysResourceCategoryEnum.BUTTON.getValue()); - List allMenuAndButtonAndFieldList = sysMenuService.list(lambdaQueryWrapper); + if(!containsTen) { + lambdaQueryWrapper.ne(SysMenu::getCode, SysBuildInEnum.BUILD_IN_NO_TEN_MENU_CODE.getValue()); + } + return this.resourceTreeSelector(sysMenuService.list(lambdaQueryWrapper)); + } + + @Override + public List resourceTreeSelector(List originDataList) { List sysModuleList = CollectionUtil.newArrayList(); List sysMenuList = CollectionUtil.newArrayList(); List sysButtonList = CollectionUtil.newArrayList(); - allMenuAndButtonAndFieldList.forEach(sysMenu -> { + originDataList.forEach(sysMenu -> { if (sysMenu.getCategory().equals(SysResourceCategoryEnum.MODULE.getValue())) { sysModuleList.add(sysMenu); } @@ -408,6 +445,11 @@ public class SysRoleServiceImpl extends ServiceImpl impl return BeanUtil.copyToList(mobileMenuApi.mobileMenuTreeSelector(), SysRoleGrantMobileMenuTreeResult.class); } + @Override + public List mobileMenuTreeSelector(List originDataList) { + return BeanUtil.copyToList(mobileMenuApi.mobileMenuTreeSelector(originDataList), SysRoleGrantMobileMenuTreeResult.class); + } + @Override public List permissionTreeSelector() { List permissionResult = CollectionUtil.newArrayList(); @@ -440,55 +482,57 @@ public class SysRoleServiceImpl extends ServiceImpl impl @Override public Page roleSelector(SysRoleSelectorRoleParam sysRoleSelectorRoleParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 - lambdaQueryWrapper.select(SysRole::getId, SysRole::getOrgId, SysRole::getName, + queryWrapper.lambda().select(SysRole::getId, SysRole::getOrgId, SysRole::getName, SysRole::getCode, SysRole::getCategory, SysRole::getSortCode); if(ObjectUtil.isNotEmpty(sysRoleSelectorRoleParam.getOrgId())) { - lambdaQueryWrapper.eq(SysRole::getOrgId, sysRoleSelectorRoleParam.getOrgId()); + queryWrapper.lambda().eq(SysRole::getOrgId, sysRoleSelectorRoleParam.getOrgId()); } if(ObjectUtil.isNotEmpty(sysRoleSelectorRoleParam.getCategory())) { - lambdaQueryWrapper.eq(SysRole::getCategory, sysRoleSelectorRoleParam.getCategory()); + queryWrapper.lambda().eq(SysRole::getCategory, sysRoleSelectorRoleParam.getCategory()); } if(ObjectUtil.isNotEmpty(sysRoleSelectorRoleParam.getSearchKey())) { - lambdaQueryWrapper.like(SysRole::getName, sysRoleSelectorRoleParam.getSearchKey()); + queryWrapper.lambda().like(SysRole::getName, sysRoleSelectorRoleParam.getSearchKey()); } if(ObjectUtil.isNotEmpty(sysRoleSelectorRoleParam.getDataScopeList())) { - lambdaQueryWrapper.in(SysRole::getOrgId, sysRoleSelectorRoleParam.getDataScopeList()); + queryWrapper.lambda().in(SysRole::getOrgId, sysRoleSelectorRoleParam.getDataScopeList()); } // 排除超管角色 if(sysRoleSelectorRoleParam.isExcludeSuperAdmin()) { - lambdaQueryWrapper.ne(SysRole::getCode, SysBuildInEnum.BUILD_IN_ROLE_CODE.getValue()); + queryWrapper.lambda().ne(SysRole::getCode, SysBuildInEnum.BUILD_IN_ROLE_CODE.getValue()); } - lambdaQueryWrapper.orderByAsc(SysRole::getSortCode); - return this.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysRole::getSortCode); + return this.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } @Override public Page userSelector(SysRoleSelectorUserParam sysRoleSelectorUserParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + // 只查询状态为正常的 + queryWrapper.lambda().eq(SysUser::getUserStatus, SysUserStatusEnum.ENABLE.getValue()); // 只查询部分字段 - lambdaQueryWrapper.select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, + queryWrapper.lambda().select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, SysUser::getName, SysUser::getSortCode, SysUser::getGender, SysUser::getEntryDate); // 如果查询条件为空,则直接查询 if(ObjectUtil.isAllEmpty(sysRoleSelectorUserParam.getOrgId(), sysRoleSelectorUserParam.getSearchKey())) { return sysUserService.getAllUserSelectorList(); } else { if (ObjectUtil.isNotEmpty(sysRoleSelectorUserParam.getOrgId())) { - // 如果组织id不为空,则查询该组织及其子极其子下的所有人 + // 如果组织id不为空,则查询该组织及其子组织下的所有人 List childOrgIdList = CollStreamUtil.toList(sysOrgService.getChildListById(sysOrgService .getAllOrgList(), sysRoleSelectorUserParam.getOrgId(), true), SysOrg::getId); if (ObjectUtil.isNotEmpty(childOrgIdList)) { - lambdaQueryWrapper.in(SysUser::getOrgId, childOrgIdList); + queryWrapper.lambda().in(SysUser::getOrgId, childOrgIdList); } else { return new Page<>(); } } if (ObjectUtil.isNotEmpty(sysRoleSelectorUserParam.getSearchKey())) { - lambdaQueryWrapper.like(SysUser::getName, sysRoleSelectorUserParam.getSearchKey()); + queryWrapper.lambda().like(SysUser::getName, sysRoleSelectorUserParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysUser::getSortCode); - return sysUserService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysUser::getSortCode); + return sysUserService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/sys/SysApiProvider.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/sys/SysApiProvider.java new file mode 100644 index 00000000..c1cf11a1 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/sys/SysApiProvider.java @@ -0,0 +1,20 @@ +package vip.xiaonuo.sys.modular.sys; + +import org.springframework.stereotype.Service; +import vip.xiaonuo.sys.api.SysApi; +import vip.xiaonuo.sys.core.util.SysPasswordUtl; + +/** + * 系统模块综合API接口实现类 + * + * @author xuyuxiang + * @date 2022/9/26 14:30 + **/ +@Service +public class SysApiProvider implements SysApi { + + @Override + public String getDefaultPassword() { + return SysPasswordUtl.getDefaultPassword(); + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserCenterController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserCenterController.java index fab0d88e..d1286ac0 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserCenterController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserCenterController.java @@ -14,11 +14,13 @@ package vip.xiaonuo.sys.modular.user.controller; import cn.dev33.satoken.stp.StpUtil; import cn.hutool.core.lang.tree.Tree; +import cn.hutool.json.JSONObject; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; -import jakarta.validation.Valid; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.*; import org.springframework.web.multipart.MultipartFile; @@ -36,15 +38,17 @@ import vip.xiaonuo.sys.modular.user.result.SysUserPicValidCodeResult; import vip.xiaonuo.sys.modular.user.result.SysUserPositionResult; import vip.xiaonuo.sys.modular.user.service.SysUserService; +import javax.validation.Valid; import java.util.List; /** - * 用户个人控制器 + * B端用户个人控制器 * * @author xuyuxiang * @date 2022/4/22 9:34 **/ -@Tag(name = "用户个人控制器") +@Tag(name = "B端用户个人控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 10) @RestController @Validated public class SysUserCenterController { @@ -58,6 +62,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/7/8 9:26 **/ + @ApiOperationSupport(order = 1) @Operation(summary = "获取图片验证码") @GetMapping("/sys/userCenter/getPicCaptcha") public CommonResult getPicCaptcha() { @@ -70,6 +75,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 2) @Operation(summary = "找回密码获取手机验证码") @GetMapping("/sys/userCenter/findPasswordGetPhoneValidCode") public CommonResult findPasswordGetPhoneValidCode(@Valid SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { @@ -82,6 +88,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 3) @Operation(summary = "找回密码获取邮箱验证码") @GetMapping("/sys/userCenter/findPasswordGetEmailValidCode") public CommonResult findPasswordGetEmailValidCode(@Valid SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { @@ -94,6 +101,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 4) @Operation(summary = "通过手机号找回用户密码") @CommonLog("通过手机号找回用户密码") @PostMapping("/sys/userCenter/findPasswordByPhone") @@ -108,6 +116,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 5) @Operation(summary = "通过邮箱找回用户密码") @CommonLog("通过邮箱找回用户密码") @PostMapping("/sys/userCenter/findPasswordByEmail") @@ -117,16 +126,155 @@ public class SysUserCenterController { } /** - * 修改用户密码 + * 修改密码获取手机验证码 * * @author xuyuxiang * @date 2021/10/13 14:01 **/ - @Operation(summary = "修改用户密码") - @CommonLog("修改用户密码") - @PostMapping("/sys/userCenter/updatePassword") - public CommonResult updatePassword(@RequestBody @Valid SysUserUpdatePwdParam sysUserUpdatePwdParam) { - sysUserService.updatePassword(sysUserUpdatePwdParam); + @ApiOperationSupport(order = 6) + @Operation(summary = "修改密码获取手机验证码") + @GetMapping("/sys/userCenter/updatePasswordGetPhoneValidCode") + public CommonResult updatePasswordGetPhoneValidCode(@Valid SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { + return CommonResult.data(sysUserService.updatePasswordGetPhoneValidCode(sysUserGetPhoneValidCodeParam)); + } + + /** + * 修改密码获取邮箱验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 7) + @Operation(summary = "修改密码获取邮箱验证码") + @GetMapping("/sys/userCenter/updatePasswordGetEmailValidCode") + public CommonResult updatePasswordGetEmailValidCode(@Valid SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { + return CommonResult.data(sysUserService.updatePasswordGetEmailValidCode(sysUserGetEmailValidCodeParam)); + } + + /** + * 通过验证旧密码修改用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 8) + @Operation(summary = "通过验证旧密码修改用户密码") + @CommonLog("通过验证旧密码修改用户密码") + @PostMapping("/sys/userCenter/updatePasswordByOld") + public CommonResult updatePasswordByOld(@RequestBody @Valid SysUserUpdatePwdByOldParam sysUserUpdatePwdByOldParam) { + sysUserService.updatePasswordByOld(sysUserUpdatePwdByOldParam); + return CommonResult.ok(); + } + + /** + * 通过验证手机号修改用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 9) + @Operation(summary = "通过验证手机号修改用户密码") + @CommonLog("通过验证手机号修改用户密码") + @PostMapping("/sys/userCenter/updatePasswordByPhone") + public CommonResult updatePasswordByPhone(@RequestBody @Valid SysUserUpdatePwdByPhoneParam sysUserUpdatePwdByPhoneParam) { + sysUserService.updatePasswordByPhone(sysUserUpdatePwdByPhoneParam); + return CommonResult.ok(); + } + + /** + * 通过验证邮箱修改用户密码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 10) + @Operation(summary = "通过验证邮箱修改用户密码") + @CommonLog("通过验证邮箱修改用户密码") + @PostMapping("/sys/userCenter/updatePasswordByEmail") + public CommonResult updatePasswordByEmail(@RequestBody @Valid SysUserUpdatePwdByEmailParam sysUserUpdatePwdByEmailParam) { + sysUserService.updatePasswordByEmail(sysUserUpdatePwdByEmailParam); + return CommonResult.ok(); + } + + /** + * 绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 11) + @Operation(summary = "绑定手机号获取手机验证码") + @GetMapping("/sys/userCenter/bindPhoneGetPhoneValidCode") + public CommonResult bindPhoneGetPhoneValidCode(@Valid SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { + return CommonResult.data(sysUserService.bindPhoneGetPhoneValidCode(sysUserGetPhoneValidCodeParam)); + } + + /** + * 修改绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 12) + @Operation(summary = "修改绑定手机号获取手机验证码") + @GetMapping("/sys/userCenter/updateBindPhoneGetPhoneValidCode") + public CommonResult updateBindPhoneGetPhoneValidCode(@Valid SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { + return CommonResult.data(sysUserService.updateBindPhoneGetPhoneValidCode(sysUserGetPhoneValidCodeParam)); + } + + /** + * 绑定手机号 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 13) + @Operation(summary = "绑定手机号") + @CommonLog("绑定手机号") + @PostMapping("/sys/userCenter/bindPhone") + public CommonResult bindPhone(@RequestBody @Valid SysUserBindPhoneParam sysUserBindPhoneParam) { + sysUserService.bindPhone(sysUserBindPhoneParam); + return CommonResult.ok(); + } + + /** + * 绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 14) + @Operation(summary = "绑定邮箱获取邮箱验证码") + @GetMapping("/sys/userCenter/bindEmailGetEmailValidCode") + public CommonResult bindEmailGetEmailValidCode(@Valid SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { + return CommonResult.data(sysUserService.bindEmailGetEmailValidCode(sysUserGetEmailValidCodeParam)); + } + + /** + * 修改绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 15) + @Operation(summary = "修改绑定邮箱获取邮箱验证码") + @GetMapping("/sys/userCenter/updateBindEmailGetEmailValidCode") + public CommonResult updateBindEmailGetEmailValidCode(@Valid SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { + return CommonResult.data(sysUserService.updateBindEmailGetEmailValidCode(sysUserGetEmailValidCodeParam)); + } + + /** + * 绑定邮箱 + * + * @author xuyuxiang + * @date 2021/10/13 14:01 + **/ + @ApiOperationSupport(order = 16) + @Operation(summary = "绑定邮箱") + @CommonLog("绑定邮箱") + @PostMapping("/sys/userCenter/bindEmail") + public CommonResult bindEmail(@RequestBody @Valid SysUserBindEmailParam sysUserBindEmailParam) { + sysUserService.bindEmail(sysUserBindEmailParam); return CommonResult.ok(); } @@ -136,6 +284,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 17) @Operation(summary = "修改用户头像") @CommonLog("修改用户头像") @PostMapping("/sys/userCenter/updateAvatar") @@ -149,6 +298,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 18) @Operation(summary = "修改用户签名图片") @CommonLog("修改用户签名图片") @PostMapping("/sys/userCenter/updateSignature") @@ -157,68 +307,13 @@ public class SysUserCenterController { return CommonResult.ok(); } - /** - * 获取登录用户的菜单(B端、PC端菜单) - * - * @author xuyuxiang - * @date 2022/4/24 20:00 - */ - @Operation(summary = "获取登录用户PC端菜单") - @GetMapping("/sys/userCenter/loginMenu") - public CommonResult>> loginMenu() { - SysUserIdParam sysUserIdParam = new SysUserIdParam(); - sysUserIdParam.setId(StpUtil.getLoginIdAsString()); - return CommonResult.data(sysUserService.ownMenu(sysUserIdParam)); - } - - /** - * 获取登录用户的菜单(B端、移动端菜单) - * - * @author xuyuxiang - * @date 2022/4/24 20:00 - */ - @Operation(summary = "获取登录用户移动端菜单") - @GetMapping("/sys/userCenter/loginMobileMenu") - public CommonResult>> loginMobileMenu() { - SysUserIdParam sysUserIdParam = new SysUserIdParam(); - sysUserIdParam.setId(StpUtil.getLoginIdAsString()); - return CommonResult.data(sysUserService.ownMobileMenu(sysUserIdParam)); - } - - /** - * 获取登录用户组织树 - * - * @author xuyuxiang - * @date 2022/4/24 20:00 - */ - @Operation(summary = "获取登录用户组织树") - @GetMapping("/sys/userCenter/loginOrgTree") - public CommonResult>> loginOrgTree() { - SysUserIdParam sysUserIdParam = new SysUserIdParam(); - sysUserIdParam.setId(StpUtil.getLoginIdAsString()); - return CommonResult.data(sysUserService.loginOrgTree(sysUserIdParam)); - } - - /** - * 获取登录用户的职位信息 - * - * @author xuyuxiang - * @date 2022/4/24 20:00 - */ - @Operation(summary = "获取登录用户的职位信息") - @GetMapping("/sys/userCenter/loginPositionInfo") - public CommonResult> loginPositionInfo() { - SysUserIdParam sysUserIdParam = new SysUserIdParam(); - sysUserIdParam.setId(StpUtil.getLoginIdAsString()); - return CommonResult.data(sysUserService.loginPositionInfo(sysUserIdParam)); - } - /** * 编辑个人信息 * * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 19) @Operation(summary = "编辑个人信息") @CommonLog("编辑个人信息") @PostMapping("/sys/userCenter/updateUserInfo") @@ -233,6 +328,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 20) @Operation(summary = "编辑个人工作台") @CommonLog("编辑个人工作台") @PostMapping("/sys/userCenter/updateUserWorkbench") @@ -241,12 +337,73 @@ public class SysUserCenterController { return CommonResult.ok(); } + /** + * 获取登录用户的菜单(B端、PC端菜单) + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 21) + @Operation(summary = "获取登录用户PC端菜单") + @GetMapping("/sys/userCenter/loginMenu") + public CommonResult>> loginMenu() { + SysUserIdParam sysUserIdParam = new SysUserIdParam(); + sysUserIdParam.setId(StpUtil.getLoginIdAsString()); + return CommonResult.data(sysUserService.ownMenu(sysUserIdParam)); + } + + /** + * 获取登录用户的菜单(B端、移动端菜单) + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 22) + @Operation(summary = "获取登录用户移动端菜单") + @GetMapping("/sys/userCenter/loginMobileMenu") + public CommonResult>> loginMobileMenu() { + SysUserIdParam sysUserIdParam = new SysUserIdParam(); + sysUserIdParam.setId(StpUtil.getLoginIdAsString()); + return CommonResult.data(sysUserService.ownMobileMenu(sysUserIdParam)); + } + + /** + * 获取登录用户组织树 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 23) + @Operation(summary = "获取登录用户组织树") + @GetMapping("/sys/userCenter/loginOrgTree") + public CommonResult>> loginOrgTree() { + SysUserIdParam sysUserIdParam = new SysUserIdParam(); + sysUserIdParam.setId(StpUtil.getLoginIdAsString()); + return CommonResult.data(sysUserService.loginOrgTree(sysUserIdParam)); + } + + /** + * 获取登录用户的职位信息 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 24) + @Operation(summary = "获取登录用户的职位信息") + @GetMapping("/sys/userCenter/loginPositionInfo") + public CommonResult> loginPositionInfo() { + SysUserIdParam sysUserIdParam = new SysUserIdParam(); + sysUserIdParam.setId(StpUtil.getLoginIdAsString()); + return CommonResult.data(sysUserService.loginPositionInfo(sysUserIdParam)); + } + /** * 获取登录用户的工作台 * * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 25) @Operation(summary = "获取登录用户的工作台") @GetMapping("/sys/userCenter/loginWorkbench") public CommonResult loginWorkbench() { @@ -261,6 +418,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 26) @Operation(summary = "获取登录用户的站内信分页") @GetMapping("/sys/userCenter/loginUnreadMessagePage") public CommonResult> loginMessagePage(SysUserMessagePageParam sysUserMessagePageParam) { @@ -273,6 +431,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 27) @Operation(summary = "读取登录用户站内信详情") @GetMapping("/sys/userCenter/loginUnreadMessageDetail") public CommonResult loginMessageDetail(@Valid SysUserMessageIdParam sysUserMessageIdParam) { @@ -285,6 +444,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 28) @Operation(summary = "根据id集合获取组织集合") @PostMapping("/sys/userCenter/getOrgListByIdList") public CommonResult> getOrgListByIdList(@RequestBody @Valid SysUserIdListParam sysUserIdListParam) { @@ -297,6 +457,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 29) @Operation(summary = "根据id集合获取用户集合") @PostMapping("/sys/userCenter/getUserListByIdList") public CommonResult> getUserListByIdList(@RequestBody @Valid SysUserIdListParam sysUserIdListParam) { @@ -309,6 +470,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 30) @Operation(summary = "根据id集合获取职位集合") @PostMapping("/sys/userCenter/getPositionListByIdList") public CommonResult> getPositionListByIdList(@RequestBody @Valid SysUserIdListParam sysUserIdListParam) { @@ -321,6 +483,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 31) @Operation(summary = "根据id集合获取角色集合") @PostMapping("/sys/userCenter/getRoleListByIdList") public CommonResult> getRoleListByIdList(@RequestBody @Valid SysUserIdListParam sysUserIdListParam) { @@ -333,6 +496,7 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 32) @Operation(summary = "根据id集合获取用户组集合") @PostMapping("/sys/userCenter/getGroupListByIdList") public CommonResult> getGroupListByIdList(@RequestBody @Valid SysUserGroupIdListParam sysUserGroupIdListParam) { @@ -345,9 +509,62 @@ public class SysUserCenterController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 33) @Operation(summary = "根据id获取头像") - @PostMapping("/sys/userCenter/getAvatarById") - public CommonResult getAvatarById(@RequestBody @javax.validation.Valid SysUserIdParam sysUserIdParam) { + @GetMapping("/sys/userCenter/getAvatarById") + public CommonResult getAvatarById(@Valid SysUserIdParam sysUserIdParam) { return CommonResult.data(sysUserService.getAvatarById(sysUserIdParam)); } + + /** + * 判断当前用户是否需要绑定手机号 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 34) + @Operation(summary = "判断当前用户是否需要绑定手机号") + @GetMapping("/sys/userCenter/isUserNeedBindPhone") + public CommonResult isUserNeedBindPhone() { + return CommonResult.data(sysUserService.isUserNeedBindPhone()); + } + + /** + * 判断当前用户是否需要绑定邮箱 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 35) + @Operation(summary = "判断当前用户是否需要绑定邮箱") + @GetMapping("/sys/userCenter/isUserNeedBindEmail") + public CommonResult isUserNeedBindEmail() { + return CommonResult.data(sysUserService.isUserNeedBindEmail()); + } + + /** + * 判断当前用户密码是否过期 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 36) + @Operation(summary = "判断当前用户密码是否过期") + @GetMapping("/sys/userCenter/isUserPasswordExpired") + public CommonResult isUserPasswordExpired() { + return CommonResult.data(sysUserService.isUserPasswordExpired()); + } + + /** + * 获取修改密码验证方式及配置 + * + * @author xuyuxiang + * @date 2022/4/24 20:00 + */ + @ApiOperationSupport(order = 37) + @Operation(summary = "获取修改密码验证方式及配置") + @GetMapping("/sys/userCenter/getUpdatePasswordValidConfig") + public CommonResult getUpdatePasswordValidConfig() { + return CommonResult.data(sysUserService.getUpdatePasswordValidConfig()); + } } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserController.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserController.java index 3faccdf2..503f69d1 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserController.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/controller/SysUserController.java @@ -15,11 +15,12 @@ package vip.xiaonuo.sys.modular.user.controller; import cn.hutool.core.lang.tree.Tree; import cn.hutool.json.JSONObject; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; +import com.github.xiaoymin.knife4j.annotations.ApiSupport; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.Resource; import jakarta.servlet.http.HttpServletResponse; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotEmpty; import org.springframework.http.MediaType; import org.springframework.validation.annotation.Validated; @@ -31,21 +32,24 @@ import vip.xiaonuo.sys.modular.org.entity.SysOrg; import vip.xiaonuo.sys.modular.position.entity.SysPosition; import vip.xiaonuo.sys.modular.role.entity.SysRole; import vip.xiaonuo.sys.modular.user.entity.SysUser; +import vip.xiaonuo.sys.modular.user.enums.SysUserSourceFromTypeEnum; import vip.xiaonuo.sys.modular.user.param.*; import vip.xiaonuo.sys.modular.user.result.SysUserOwnPermissionResult; import vip.xiaonuo.sys.modular.user.result.SysUserOwnResourceResult; import vip.xiaonuo.sys.modular.user.service.SysUserService; +import javax.validation.Valid; import java.io.IOException; import java.util.List; /** - * 用户控制器 + * B端用户控制器 * * @author xuyuxiang * @date 2022/4/22 9:34 **/ -@Tag(name = "用户控制器") +@Tag(name = "B端用户控制器") +@ApiSupport(author = "SNOWY_TEAM", order = 9) @RestController @Validated public class SysUserController { @@ -59,6 +63,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 1) @Operation(summary = "获取用户分页") @GetMapping("/sys/user/page") public CommonResult> page(SysUserPageParam sysUserPageParam) { @@ -71,11 +76,12 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 2) @Operation(summary = "添加用户") @CommonLog("添加用户") @PostMapping("/sys/user/add") public CommonResult add(@RequestBody @Valid SysUserAddParam sysUserAddParam) { - sysUserService.add(sysUserAddParam); + sysUserService.add(sysUserAddParam, SysUserSourceFromTypeEnum.SYSTEM_ADD.getValue()); return CommonResult.ok(); } @@ -85,6 +91,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:47 */ + @ApiOperationSupport(order = 3) @Operation(summary = "编辑用户") @CommonLog("编辑用户") @PostMapping("/sys/user/edit") @@ -99,11 +106,12 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 4) @Operation(summary = "删除用户") @CommonLog("删除用户") @PostMapping("/sys/user/delete") public CommonResult delete(@RequestBody @Valid @NotEmpty(message = "集合不能为空") - List sysUserIdParamList) { + List sysUserIdParamList) { sysUserService.delete(sysUserIdParamList); return CommonResult.ok(); } @@ -114,6 +122,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 5) @Operation(summary = "获取用户详情") @GetMapping("/sys/user/detail") public CommonResult detail(@Valid SysUserIdParam sysUserIdParam) { @@ -126,6 +135,7 @@ public class SysUserController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 6) @Operation(summary = "禁用用户") @CommonLog("禁用用户") @PostMapping("/sys/user/disableUser") @@ -140,6 +150,7 @@ public class SysUserController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 7) @Operation(summary = "启用用户") @CommonLog("启用用户") @PostMapping("/sys/user/enableUser") @@ -154,6 +165,7 @@ public class SysUserController { * @author xuyuxiang * @date 2021/10/13 14:01 **/ + @ApiOperationSupport(order = 8) @Operation(summary = "重置用户密码") @CommonLog("重置用户密码") @PostMapping("/sys/user/resetPassword") @@ -168,6 +180,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 9) @Operation(summary = "获取用户拥有角色") @GetMapping("/sys/user/ownRole") public CommonResult> ownRole(@Valid SysUserIdParam sysUserIdParam) { @@ -180,6 +193,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 10) @Operation(summary = "给用户授权角色") @CommonLog("给用户授权角色") @PostMapping("/sys/user/grantRole") @@ -194,6 +208,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 11) @Operation(summary = "获取用户拥有资源") @GetMapping("/sys/user/ownResource") public CommonResult ownResource(@Valid SysUserIdParam sysUserIdParam) { @@ -206,6 +221,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 12) @Operation(summary = "给用户授权资源") @CommonLog("给用户授权资源") @PostMapping("/sys/user/grantResource") @@ -220,6 +236,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 13) @Operation(summary = "获取用户拥有权限") @GetMapping("/sys/user/ownPermission") public CommonResult ownPermission(@Valid SysUserIdParam sysUserIdParam) { @@ -232,6 +249,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 14) @Operation(summary = "给用户授权权限") @CommonLog("给用户授权权限") @PostMapping("/sys/user/grantPermission") @@ -246,6 +264,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 15) @Operation(summary = "下载用户导入模板") @CommonLog("下载用户导入模板") @GetMapping(value = "/sys/user/downloadImportUserTemplate", produces = MediaType.APPLICATION_OCTET_STREAM_VALUE) @@ -259,6 +278,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 16) @Operation(summary = "用户导入") @CommonLog("用户导入") @PostMapping("/sys/user/import") @@ -272,6 +292,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 17) @Operation(summary = "用户导出") @CommonLog("用户导出") @GetMapping(value = "/sys/user/export", produces = MediaType.APPLICATION_OCTET_STREAM_VALUE) @@ -285,6 +306,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 18) @Operation(summary = "导出用户个人信息") @CommonLog("导出用户个人信息") @GetMapping(value = "/sys/user/exportUserInfo", produces = MediaType.APPLICATION_OCTET_STREAM_VALUE) @@ -300,6 +322,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 19) @Operation(summary = "获取组织树选择器") @GetMapping("/sys/user/orgTreeSelector") public CommonResult>> orgTreeSelector() { @@ -312,6 +335,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 20) @Operation(summary = "获取组织列表选择器") @GetMapping("/sys/user/orgListSelector") public CommonResult> orgListSelector(SysUserSelectorOrgListParam sysUserSelectorOrgListParam) { @@ -324,6 +348,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 21) @Operation(summary = "获取职位选择器") @GetMapping("/sys/user/positionSelector") public CommonResult> positionSelector(SysUserSelectorPositionParam sysUserSelectorPositionParam) { @@ -336,6 +361,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 22) @Operation(summary = "获取角色选择器") @GetMapping("/sys/user/roleSelector") public CommonResult> roleSelector(SysUserSelectorRoleParam sysUserSelectorRoleParam) { @@ -348,6 +374,7 @@ public class SysUserController { * @author xuyuxiang * @date 2022/4/24 20:00 */ + @ApiOperationSupport(order = 23) @Operation(summary = "获取用户选择器") @GetMapping("/sys/user/userSelector") public CommonResult> userSelector(SysUserSelectorUserParam sysUserSelectorUserParam) { diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUser.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUser.java index 6c0cf78c..b5b373ed 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUser.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUser.java @@ -47,12 +47,12 @@ public class SysUser extends CommonEntity { /** 头像 */ @Schema(description = "头像") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String avatar; /** 签名 */ @Schema(description = "签名") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String signature; /** 账号 */ @@ -70,156 +70,156 @@ public class SysUser extends CommonEntity { /** 昵称 */ @Schema(description = "昵称") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nickname; /** 性别 */ @Schema(description = "性别") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) - @Trans(type = TransType.DICTIONARY, key = "GENDER") + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) + @Trans(type = TransType.DICTIONARY,key = "GENDER") private String gender; /** 年龄 */ @Schema(description = "年龄") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String age; /** 出生日期 */ @Schema(description = "出生日期") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String birthday; /** 民族 */ @Schema(description = "民族") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nation; /** 籍贯 */ @Schema(description = "籍贯") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String nativePlace; /** 家庭住址 */ @Schema(description = "家庭住址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String homeAddress; /** 通信地址 */ @Schema(description = "通信地址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String mailingAddress; /** 证件类型 */ @Schema(description = "证件类型") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String idCardType; /** 证件号码 */ @Schema(description = "证件号码") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String idCardNumber; /** 文化程度 */ @Schema(description = "文化程度") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String cultureLevel; /** 政治面貌 */ @Schema(description = "政治面貌") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String politicalOutlook; /** 毕业院校 */ @Schema(description = "毕业院校") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String college; /** 学历 */ @Schema(description = "学历") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String education; /** 学制 */ @Schema(description = "学制") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String eduLength; /** 学位 */ @Schema(description = "学位") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String degree; /** 手机 */ @Schema(description = "手机") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String phone; /** 邮箱 */ @Schema(description = "邮箱") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String email; /** 家庭电话 */ @Schema(description = "家庭电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String homeTel; /** 办公电话 */ @Schema(description = "办公电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String officeTel; /** 紧急联系人 */ @Schema(description = "紧急联系人") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String emergencyContact; /** 紧急联系人电话 */ @Schema(description = "紧急联系人电话") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED, typeHandler = CommonSm4CbcTypeHandler.class) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS, typeHandler = CommonSm4CbcTypeHandler.class) private String emergencyPhone; /** 紧急联系人地址 */ @Schema(description = "紧急联系人地址") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String emergencyAddress; /** 员工编号 */ @Schema(description = "员工编号") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String empNo; /** 入职日期 */ @Schema(description = "入职日期") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String entryDate; /** 组织id */ @Schema(description = "组织id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.SIMPLE, target = SysOrg.class, fields = "name", alias = "org", ref = "orgName") private String orgId; /** 职位id */ @Schema(description = "职位id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.SIMPLE, target = SysPosition.class, fields = "name", alias = "position", ref = "positionName") private String positionId; /** 职级 */ @Schema(description = "职级") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String positionLevel; /** 主管id */ @Schema(description = "主管id") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) @Trans(type = TransType.SIMPLE, target = SysUser.class, fields = "name", alias = "director", ref = "directorName") private String directorId; /** 兼任信息 */ @Schema(description = "兼任信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String positionJson; /** 上次登录ip */ @@ -264,7 +264,7 @@ public class SysUser extends CommonEntity { /** 扩展信息 */ @Schema(description = "扩展信息") - @TableField(insertStrategy = FieldStrategy.IGNORED, updateStrategy = FieldStrategy.IGNORED) + @TableField(insertStrategy = FieldStrategy.ALWAYS, updateStrategy = FieldStrategy.ALWAYS) private String extJson; /** 组织名称 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUserExt.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUserExt.java new file mode 100644 index 00000000..667da98e --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUserExt.java @@ -0,0 +1,52 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +import java.util.Date; + +/** + * 用户扩展实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("SYS_USER_EXT") +public class SysUserExt extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 用户id */ + @Schema(description = "用户id") + private String userId; + + /** 来源类别 */ + @Schema(description = "来源类别") + private String sourceFromType; + + /** 密码修改日期 */ + @Schema(description = "密码修改日期") + private Date passwordUpdateTime; + +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUserPassword.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUserPassword.java new file mode 100644 index 00000000..602e3b59 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/entity/SysUserPassword.java @@ -0,0 +1,46 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.entity; + +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.Getter; +import lombok.Setter; +import vip.xiaonuo.common.pojo.CommonEntity; + +/** + * 用户密码实体 + * + * @author xuyuxiang + * @date 2022/4/21 16:13 + **/ +@Getter +@Setter +@TableName("SYS_USER_PASSWORD") +public class SysUserPassword extends CommonEntity { + + /** id */ + @TableId + @Schema(description = "id") + private String id; + + /** 用户id */ + @Schema(description = "用户id") + private String userId; + + /** 密码 */ + @Schema(description = "密码") + private String password; + +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysEmailEngineTypeEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysEmailEngineTypeEnum.java new file mode 100644 index 00000000..8ad78a75 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysEmailEngineTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.enums; + +import lombok.Getter; + +/** + * 邮件发送引擎类型枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum SysEmailEngineTypeEnum { + + /** 本地 */ + LOCAL("LOCAL"), + + /** 阿里云 */ + ALIYUN("ALIYUN"), + + /** 腾讯云 */ + TENCENT("TENCENT"); + + private final String value; + + SysEmailEngineTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysSmsEngineTypeEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysSmsEngineTypeEnum.java new file mode 100644 index 00000000..34b0cb30 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysSmsEngineTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.enums; + +import lombok.Getter; + +/** + * 短信发送引擎类型枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum SysSmsEngineTypeEnum { + + /** 阿里云 */ + ALIYUN("ALIYUN"), + + /** 腾讯云 */ + TENCENT("TENCENT"), + + /** 小诺短信 */ + XIAONUO("XIAONUO"); + + private final String value; + + SysSmsEngineTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysUpdatePasswordValidTypeEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysUpdatePasswordValidTypeEnum.java new file mode 100644 index 00000000..a2a307be --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysUpdatePasswordValidTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.enums; + +import lombok.Getter; + +/** + * 系统修改密码验证方式枚举 + * + * @author xuyuxiang + * @date 2022/6/16 16:14 + **/ +@Getter +public enum SysUpdatePasswordValidTypeEnum { + + /** 旧密码 */ + OLD("OLD"), + + /** 手机号 */ + PHONE("PHONE"), + + /** 邮箱 */ + EMAIL("EMAIL"); + + private final String value; + + SysUpdatePasswordValidTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysUserSourceFromTypeEnum.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysUserSourceFromTypeEnum.java new file mode 100644 index 00000000..e5bfda34 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/enums/SysUserSourceFromTypeEnum.java @@ -0,0 +1,40 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.enums; + +import lombok.Getter; + +/** + * 用户来源类型枚举 + * + * @author xuyuxiang + * @date 2022/4/21 19:56 + **/ +@Getter +public enum SysUserSourceFromTypeEnum { + + /** 系统自建 */ + SYSTEM_ADD("SYSTEM_ADD"), + + /** 用户注册 */ + SYSTEM_REGISTER("SYSTEM_REGISTER"), + + /** 身份源 */ + ID_SOURCE("ID_SOURCE"); + + private final String value; + + SysUserSourceFromTypeEnum(String value) { + this.value = value; + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserExtMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserExtMapper.java new file mode 100644 index 00000000..acb4cf03 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserExtMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.sys.modular.user.entity.SysUserExt; + +/** + * 用户扩展Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface SysUserExtMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserMapper.java index 8ba5affb..20d38c1c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserMapper.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserMapper.java @@ -12,10 +12,7 @@ */ package vip.xiaonuo.sys.modular.user.mapper; -import com.baomidou.mybatisplus.annotation.InterceptorIgnore; -import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import org.apache.ibatis.annotations.Param; import vip.xiaonuo.sys.modular.user.entity.SysUser; /** @@ -26,12 +23,4 @@ import vip.xiaonuo.sys.modular.user.entity.SysUser; **/ public interface SysUserMapper extends BaseMapper { - /** - * 删除数据并忽略插件(逻辑删除、租户拼接) - * - * @author xuyuxiang - * @date 2023/12/25 23:20 - */ - @InterceptorIgnore(tenantLine = "true") - void deleteIgnoreInterceptor(@Param("ew") LambdaQueryWrapper lambdaQueryWrapper); } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserPasswordMapper.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserPasswordMapper.java new file mode 100644 index 00000000..fc8914c0 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/SysUserPasswordMapper.java @@ -0,0 +1,25 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import vip.xiaonuo.sys.modular.user.entity.SysUserPassword; + +/** + * 用户密码Mapper接口 + * + * @author xuyuxiang + * @date 2022/4/21 18:37 + **/ +public interface SysUserPasswordMapper extends BaseMapper { +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserExtMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserExtMapper.xml new file mode 100644 index 00000000..3311e0db --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserExtMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserMapper.xml index 417cdb16..7f058c62 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserMapper.xml +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserMapper.xml @@ -2,8 +2,4 @@ - - - DELETE FROM SYS_USER ${ew.customSqlSegment} - - \ No newline at end of file + diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserPasswordMapper.xml b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserPasswordMapper.xml new file mode 100644 index 00000000..7bec95ba --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/mapper/mapping/SysUserPasswordMapper.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserAddParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserAddParam.java index 89a11c11..c5319ead 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserAddParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserAddParam.java @@ -28,22 +28,22 @@ import lombok.Setter; public class SysUserAddParam { /** 账号 */ - @Schema(description = "账号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "账号") @NotBlank(message = "account不能为空") private String account; /** 姓名 */ - @Schema(description = "姓名", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "姓名") @NotBlank(message = "name不能为空") private String name; /** 组织id */ - @Schema(description = "组织id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "组织id") @NotBlank(message = "orgId不能为空") private String orgId; /** 职位id */ - @Schema(description = "职位id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "职位id") @NotBlank(message = "positionId不能为空") private String positionId; @@ -55,6 +55,10 @@ public class SysUserAddParam { @Schema(description = "主管id") private String directorId; + /** 密码 */ + @Schema(description = "密码") + private String password; + /** 头像 */ @Schema(description = "头像,图片base64") private String avatar; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserBindEmailParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserBindEmailParam.java new file mode 100644 index 00000000..cb67619b --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserBindEmailParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 绑定邮箱参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class SysUserBindEmailParam { + + /** 邮箱 */ + @Schema(description = "邮箱") + @NotBlank(message = "email不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserBindPhoneParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserBindPhoneParam.java new file mode 100644 index 00000000..5584bada --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserBindPhoneParam.java @@ -0,0 +1,44 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 绑定手机号参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class SysUserBindPhoneParam { + + /** 手机号 */ + @Schema(description = "手机号") + @NotBlank(message = "phone不能为空") + private String phone; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserEditParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserEditParam.java index 86bf26b9..c9986267 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserEditParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserEditParam.java @@ -28,27 +28,27 @@ import lombok.Setter; public class SysUserEditParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 账号 */ - @Schema(description = "账号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "账号") @NotBlank(message = "account不能为空") private String account; /** 姓名 */ - @Schema(description = "姓名", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "姓名") @NotBlank(message = "name不能为空") private String name; /** 组织id */ - @Schema(description = "组织id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "组织id") @NotBlank(message = "orgId不能为空") private String orgId; /** 职位id */ - @Schema(description = "职位id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "职位id") @NotBlank(message = "positionId不能为空") private String positionId; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByEmailParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByEmailParam.java index 279e9220..90887620 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByEmailParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByEmailParam.java @@ -18,7 +18,7 @@ import lombok.Getter; import lombok.Setter; /** - * 用户找回密码参数 + * 用户通过邮箱找回密码参数 * * @author xuyuxiang * @date 2022/7/26 16:04 @@ -28,22 +28,22 @@ import lombok.Setter; public class SysUserFindPwdByEmailParam { /** 邮箱 */ - @Schema(description = "邮箱", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮箱") @NotBlank(message = "email不能为空") private String email; /** 验证码 */ - @Schema(description = "验证码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码") @NotBlank(message = "validCode不能为空") private String validCode; /** 验证码请求号 */ - @Schema(description = "验证码请求号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码请求号") @NotBlank(message = "validCodeReqNo不能为空") private String validCodeReqNo; /** 新密码 */ - @Schema(description = "新密码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "新密码") @NotBlank(message = "newPassword不能为空") private String newPassword; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByPhoneParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByPhoneParam.java index f4f9f6e9..d41a60f3 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByPhoneParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserFindPwdByPhoneParam.java @@ -18,7 +18,7 @@ import lombok.Getter; import lombok.Setter; /** - * 用户找回密码参数 + * 用户通过手机号找回密码参数 * * @author xuyuxiang * @date 2022/7/26 16:04 @@ -28,22 +28,22 @@ import lombok.Setter; public class SysUserFindPwdByPhoneParam { /** 手机号 */ - @Schema(description = "手机号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "手机号") @NotBlank(message = "phone不能为空") private String phone; /** 验证码 */ - @Schema(description = "验证码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码") @NotBlank(message = "validCode不能为空") private String validCode; /** 验证码请求号 */ - @Schema(description = "验证码请求号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码请求号") @NotBlank(message = "validCodeReqNo不能为空") private String validCodeReqNo; /** 新密码 */ - @Schema(description = "新密码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "新密码") @NotBlank(message = "newPassword不能为空") private String newPassword; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetEmailValidCodeParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetEmailValidCodeParam.java index 840afff6..89e22463 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetEmailValidCodeParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetEmailValidCodeParam.java @@ -28,17 +28,17 @@ import lombok.Setter; public class SysUserGetEmailValidCodeParam { /** 邮箱 */ - @Schema(description = "邮箱", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "邮箱") @NotBlank(message = "邮箱不能为空") private String email; /** 验证码 */ - @Schema(description = "验证码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码") @NotBlank(message = "验证码不能为空") private String validCode; /** 验证码请求号 */ - @Schema(description = "验证码请求号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码请求号") @NotBlank(message = "验证码请求号不能为空") private String validCodeReqNo; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetPhoneValidCodeParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetPhoneValidCodeParam.java index 14f2b9bb..473f0967 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetPhoneValidCodeParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGetPhoneValidCodeParam.java @@ -28,17 +28,17 @@ import lombok.Setter; public class SysUserGetPhoneValidCodeParam { /** 手机号 */ - @Schema(description = "手机号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "手机号") @NotBlank(message = "手机号不能为空") private String phone; /** 验证码 */ - @Schema(description = "验证码", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码") @NotBlank(message = "验证码不能为空") private String validCode; /** 验证码请求号 */ - @Schema(description = "验证码请求号", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "验证码请求号") @NotBlank(message = "验证码请求号不能为空") private String validCodeReqNo; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantPermissionParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantPermissionParam.java index c5fa21ff..8ada201f 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantPermissionParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantPermissionParam.java @@ -13,12 +13,12 @@ package vip.xiaonuo.sys.modular.user.param; import io.swagger.v3.oas.annotations.media.Schema; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; import lombok.Getter; import lombok.Setter; +import javax.validation.Valid; import java.util.List; /** @@ -32,13 +32,13 @@ import java.util.List; public class SysUserGrantPermissionParam { /** 用户id */ - @Schema(description = "用户id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "用户id") @NotBlank(message = "id不能为空") private String id; /** 授权权限信息 */ @Valid - @Schema(description = "授权权限信息", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "授权权限信息") @NotNull(message = "grantInfoList不能为空") private List grantInfoList; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantResourceParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantResourceParam.java index 023d58e8..8f934c4c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantResourceParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantResourceParam.java @@ -13,12 +13,12 @@ package vip.xiaonuo.sys.modular.user.param; import io.swagger.v3.oas.annotations.media.Schema; -import jakarta.validation.Valid; import jakarta.validation.constraints.NotBlank; import jakarta.validation.constraints.NotNull; import lombok.Getter; import lombok.Setter; +import javax.validation.Valid; import java.util.List; /** @@ -32,13 +32,13 @@ import java.util.List; public class SysUserGrantResourceParam { /** 用户id */ - @Schema(description = "用户id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "用户id") @NotBlank(message = "id不能为空") private String id; /** 授权资源信息 */ @Valid - @Schema(description = "授权资源信息", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "授权资源信息") @NotNull(message = "grantInfoList不能为空") private List grantInfoList; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantRoleParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantRoleParam.java index ca6aab17..769f40d7 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantRoleParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserGrantRoleParam.java @@ -31,12 +31,12 @@ import java.util.List; public class SysUserGrantRoleParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 角色id集合 */ - @Schema(description = "角色id集合", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "角色id集合") @NotNull(message = "roleIdList不能为空") private List roleIdList; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdListParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdListParam.java index 734a5b8f..2b72c6c6 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdListParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdListParam.java @@ -30,7 +30,7 @@ import java.util.List; public class SysUserIdListParam { /** id集合 */ - @Schema(description = "id集合", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id集合") @NotNull(message = "idList不能为空") private List idList; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdParam.java index b8d597ac..4fc94274 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysUserIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserMessageIdParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserMessageIdParam.java index cc280ab6..7ca6fb56 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserMessageIdParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserMessageIdParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysUserMessageIdParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserSignatureParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserSignatureParam.java index 0abb6892..7d8a5689 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserSignatureParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserSignatureParam.java @@ -18,7 +18,7 @@ import lombok.Getter; import lombok.Setter; /** - * 用户修改签名图片接口 + * 用户修改签名图片参数 * * @author yubaoshan * @date 2022/9/7 23:12 @@ -28,7 +28,7 @@ import lombok.Setter; public class SysUserSignatureParam { /** 签名图片base64编码 */ - @Schema(description = "signature", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "signature") @NotBlank(message = "signature签名图片不能为空") private String signature; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateInfoParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateInfoParam.java index 6f8ded9d..e33e01d1 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateInfoParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateInfoParam.java @@ -18,7 +18,7 @@ import lombok.Getter; import lombok.Setter; /** - * 编辑个人信息参数 + * 用户编辑个人信息参数 * * @author xuyuxiang * @date 2022/7/27 17:08 @@ -28,19 +28,15 @@ import lombok.Setter; public class SysUserUpdateInfoParam { /** id */ - @Schema(description = "id", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "id") @NotBlank(message = "id不能为空") private String id; /** 姓名 */ - @Schema(description = "姓名", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "姓名") @NotBlank(message = "name不能为空") private String name; - /** 手机 */ - @Schema(description = "手机") - private String phone; - /** 昵称 */ @Schema(description = "昵称") private String nickname; @@ -53,10 +49,6 @@ public class SysUserUpdateInfoParam { @Schema(description = "出生日期") private String birthday; - /** 邮箱 */ - @Schema(description = "邮箱") - private String email; - /** 签名 */ @Schema(description = "签名,图片base64") private String signature; diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByEmailParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByEmailParam.java new file mode 100644 index 00000000..40cf7b49 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByEmailParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户通过验证邮箱修改密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class SysUserUpdatePwdByEmailParam { + + /** 邮箱 */ + @Schema(description = "邮箱") + @NotBlank(message = "email不能为空") + private String email; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByOldParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByOldParam.java new file mode 100644 index 00000000..5e10b61e --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByOldParam.java @@ -0,0 +1,39 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户通过验证旧密码修改密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class SysUserUpdatePwdByOldParam { + + /** 旧密码 */ + @Schema(description = "旧密码") + @NotBlank(message = "password不能为空") + private String password; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByPhoneParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByPhoneParam.java new file mode 100644 index 00000000..ff04eb76 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdatePwdByPhoneParam.java @@ -0,0 +1,49 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.param; + +import io.swagger.v3.oas.annotations.media.Schema; +import jakarta.validation.constraints.NotBlank; +import lombok.Getter; +import lombok.Setter; + +/** + * 用户通过验证手机号修改密码参数 + * + * @author xuyuxiang + * @date 2022/7/26 16:04 + **/ +@Getter +@Setter +public class SysUserUpdatePwdByPhoneParam { + + /** 手机号 */ + @Schema(description = "手机号") + @NotBlank(message = "phone不能为空") + private String phone; + + /** 验证码 */ + @Schema(description = "验证码") + @NotBlank(message = "validCode不能为空") + private String validCode; + + /** 验证码请求号 */ + @Schema(description = "验证码请求号") + @NotBlank(message = "validCodeReqNo不能为空") + private String validCodeReqNo; + + /** 新密码 */ + @Schema(description = "新密码") + @NotBlank(message = "newPassword不能为空") + private String newPassword; +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateWorkbenchParam.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateWorkbenchParam.java index 7c097633..9c252f2f 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateWorkbenchParam.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/param/SysUserUpdateWorkbenchParam.java @@ -28,7 +28,7 @@ import lombok.Setter; public class SysUserUpdateWorkbenchParam { /** 工作台数据 */ - @Schema(description = "工作台数据", requiredMode = Schema.RequiredMode.REQUIRED) + @Schema(description = "工作台数据") @NotBlank(message = "workbenchData不能为空") private String workbenchData; } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/provider/SysLoginUserApiProvider.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/provider/SysLoginUserApiProvider.java index ed15660a..cece921f 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/provider/SysLoginUserApiProvider.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/provider/SysLoginUserApiProvider.java @@ -12,6 +12,7 @@ */ package vip.xiaonuo.sys.modular.user.provider; +import cn.hutool.core.bean.BeanUtil; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; import jakarta.annotation.Resource; @@ -19,6 +20,7 @@ import org.springframework.stereotype.Service; import vip.xiaonuo.auth.api.SaBaseLoginUserApi; import vip.xiaonuo.auth.core.pojo.SaBaseClientLoginUser; import vip.xiaonuo.auth.core.pojo.SaBaseLoginUser; +import vip.xiaonuo.sys.modular.user.entity.SysUser; import vip.xiaonuo.sys.modular.user.result.SysLoginUser; import vip.xiaonuo.sys.modular.user.service.SysUserService; @@ -92,6 +94,11 @@ public class SysLoginUserApiProvider implements SaBaseLoginUserApi { return sysUserService.getUserByPhone(phone); } + @Override + public SaBaseLoginUser getUserByEmail(String email) { + return sysUserService.getUserByEmail(email); + } + /** * 不实现C端用户信息 * @@ -103,6 +110,11 @@ public class SysLoginUserApiProvider implements SaBaseLoginUserApi { return null; } + @Override + public SaBaseClientLoginUser getClientUserByEmail(String email) { + return null; + } + /** * 根据用户id获取用户集合 * @@ -168,4 +180,31 @@ public class SysLoginUserApiProvider implements SaBaseLoginUserApi { public void updateUserLoginInfo(String userId, String device) { sysUserService.updateUserLoginInfo(userId, device); } -} \ No newline at end of file + + @Override + public SaBaseLoginUser createUserWithPhone(String phone) { + SysUser sysUser = sysUserService.createUserWithPhone(phone); + return BeanUtil.copyProperties(sysUser, SysLoginUser.class); + } + + @Override + public SaBaseClientLoginUser createClientUserWithPhone(String phone) { + return null; + } + + @Override + public SaBaseLoginUser createUserWithEmail(String email) { + SysUser sysUser = sysUserService.createUserWithEmail(email); + return BeanUtil.copyProperties(sysUser, SysLoginUser.class); + } + + @Override + public SaBaseClientLoginUser createClientUserWithEmail(String email) { + return null; + } + + @Override + public void doRegister(String account, String password) { + sysUserService.doRegister(account, password); + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageDetailResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageDetailResult.java index 32f320ff..c8af7d21 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageDetailResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageDetailResult.java @@ -29,7 +29,7 @@ import java.util.List; public class SysUserMessageDetailResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 分类 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageResult.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageResult.java index a4ed0840..e3406989 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageResult.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/result/SysUserMessageResult.java @@ -29,7 +29,7 @@ import java.util.Date; public class SysUserMessageResult { /** id */ - @Schema(description = "id") + @Schema(description = "主键") private String id; /** 分类 */ diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserExtService.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserExtService.java new file mode 100644 index 00000000..27ac58aa --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserExtService.java @@ -0,0 +1,41 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.sys.modular.user.entity.SysUserExt; + +/** + * 用户扩展Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface SysUserExtService extends IService { + + /** + * 更新用户最新修改密码时间 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void updatePasswordLastTime(String userId); + + /** + * 插入扩展信息 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void createExtInfo(String userId, String sourceFromType); +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserPasswordService.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserPasswordService.java new file mode 100644 index 00000000..300b0536 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserPasswordService.java @@ -0,0 +1,43 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import vip.xiaonuo.sys.modular.user.entity.SysUserPassword; + +import java.util.List; + +/** + * 用户密码Service接口 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +public interface SysUserPasswordService extends IService { + + /** + * 追加用户历史密码信息 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + void insertUserPasswordHistory(String userId, String newPassword); + + /** + * 获取用户前N个历史密码 + * + * @author xuyuxiang + * @date 2022/4/27 21:38 + */ + List getUserPasswordHistoryLimit(String userId, int limitValue); +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserService.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserService.java index 6738de1d..9ea8a15c 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserService.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/SysUserService.java @@ -83,7 +83,7 @@ public interface SysUserService extends IService { * @author xuyuxiang * @date 2022/4/24 20:48 */ - void add(SysUserAddParam sysUserAddParam); + void add(SysUserAddParam sysUserAddParam, String sourceFromType); /** * 编辑用户 @@ -182,12 +182,92 @@ public interface SysUserService extends IService { void findPasswordByEmail(SysUserFindPwdByEmailParam sysUserFindPwdByEmailParam); /** - * 修改用户密码 + * 修改密码获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updatePasswordGetPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam); + + /** + * 修改密码获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updatePasswordGetEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam); + + /** + * 通过验证旧密码修改用户密码 * * @author xuyuxiang * @date 2022/4/22 15:53 **/ - void updatePassword(SysUserUpdatePwdParam sysUserUpdatePwdParam); + void updatePasswordByOld(SysUserUpdatePwdByOldParam sysUserUpdatePwdByOldParam); + + /** + * 通过验证手机号修改用户密码 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + void updatePasswordByPhone(SysUserUpdatePwdByPhoneParam sysUserUpdatePwdByPhoneParam); + + /** + * 通过验证邮箱修改用户密码 + * + * @author xuyuxiang + * @date 2022/4/22 15:53 + **/ + void updatePasswordByEmail(SysUserUpdatePwdByEmailParam sysUserUpdatePwdByEmailParam); + + /** + * 绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String bindPhoneGetPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam); + + /** + * 修改绑定手机号获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updateBindPhoneGetPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam); + + /** + * 绑定手机号 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void bindPhone(SysUserBindPhoneParam sysUserBindPhoneParam); + + /** + * 绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String bindEmailGetEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam); + + /** + * 修改绑定邮箱获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + String updateBindEmailGetEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam); + + /** + * 绑定邮箱 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void bindEmail(SysUserBindEmailParam sysUserBindEmailParam); /** * 修改用户头像返回base64 @@ -471,14 +551,6 @@ public interface SysUserService extends IService { **/ List getPositionListByIdList(SysUserIdListParam sysUserIdListParam); - /** - * 根据id集合获取用户组集合 - * - * @author yubaoshan - * @date 2025/1/12 02:36 - */ - List getGroupListByIdList(SysUserGroupIdListParam sysUserGroupIdListParam); - /** * 根据id集合获取角色集合 * @@ -487,6 +559,14 @@ public interface SysUserService extends IService { **/ List getRoleListByIdList(SysUserIdListParam sysUserIdListParam); + /** + * 根据id集合获取用户组集合 + * + * @author yubaoshan + * @date 2025/1/12 02:36 + */ + List getGroupListByIdList(SysUserGroupIdListParam sysUserGroupIdListParam); + /** * 根据id获取头像 * @@ -494,4 +574,76 @@ public interface SysUserService extends IService { * @date 2023/8/28 10:10 **/ String getAvatarById(SysUserIdParam sysUserIdParam); + + /** + * 根据手机号创建用户 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + SysUser createUserWithPhone(String phone); + + /** + * 根据邮箱创建用户 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + SysUser createUserWithEmail(String email); + + /** + * 根据账号密码创建用户 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + SysUser createUserWithAccount(String account, String password); + + /** + * 判断当前用户是否需要绑定手机号 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + Boolean isUserNeedBindPhone(); + + /** + * 判断当前用户是否需要绑定邮箱 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + Boolean isUserNeedBindEmail(); + + /** + * 判断当前用户密码是否过期 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + Boolean isUserPasswordExpired(); + + /** + * 通知用户密码即将到期 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void noticeUserPasswordAboutToExpired(); + + /** + * 执行注册 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + void doRegister(String account, String password); + + /** + * 获取修改密码验证方式及配置 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + JSONObject getUpdatePasswordValidConfig(); } diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserExtServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserExtServiceImpl.java new file mode 100644 index 00000000..1722c8e2 --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserExtServiceImpl.java @@ -0,0 +1,55 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.service.impl; + +import cn.hutool.core.date.DateTime; +import cn.hutool.core.util.ObjectUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import vip.xiaonuo.sys.modular.user.entity.SysUserExt; +import vip.xiaonuo.sys.modular.user.mapper.SysUserExtMapper; +import vip.xiaonuo.sys.modular.user.service.SysUserExtService; + +/** + * 用户扩展Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class SysUserExtServiceImpl extends ServiceImpl implements SysUserExtService { + + @Override + public void updatePasswordLastTime(String userId) { + SysUserExt sysUserExt = this.getOne(new LambdaQueryWrapper().eq(SysUserExt::getUserId, userId)); + if(ObjectUtil.isEmpty(sysUserExt)){ + sysUserExt = new SysUserExt(); + sysUserExt.setUserId(userId); + sysUserExt.setPasswordUpdateTime(DateTime.now()); + this.save(sysUserExt); + } else { + sysUserExt.setPasswordUpdateTime(DateTime.now()); + this.updateById(sysUserExt); + } + } + + @Override + public void createExtInfo(String userId, String sourceFromType) { + SysUserExt sysUserExt = new SysUserExt(); + sysUserExt.setUserId(userId); + sysUserExt.setSourceFromType(sourceFromType); + sysUserExt.setPasswordUpdateTime(DateTime.now()); + this.save(sysUserExt); + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserPasswordServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserPasswordServiceImpl.java new file mode 100644 index 00000000..1cbc776c --- /dev/null +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserPasswordServiceImpl.java @@ -0,0 +1,50 @@ +/* + * Copyright [2022] [https://www.xiaonuo.vip] + * + * Snowy采用APACHE LICENSE 2.0开源协议,您在使用过程中,需要注意以下几点: + * + * 1.请不要删除和修改根目录下的LICENSE文件。 + * 2.请不要删除和修改Snowy源码头部的版权声明。 + * 3.本项目代码可免费商业使用,商业使用请保留源码和相关描述文件的项目出处,作者声明等。 + * 4.分发源码时候,请注明软件出处 https://www.xiaonuo.vip + * 5.不可二次分发开源参与同类竞品,如有想法可联系团队xiaonuobase@qq.com商议合作。 + * 6.若您的项目无法满足以上几点,需要更多功能代码,获取Snowy商业授权许可,请在官网购买授权,地址为 https://www.xiaonuo.vip + */ +package vip.xiaonuo.sys.modular.user.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.stereotype.Service; +import vip.xiaonuo.common.util.CommonCryptogramUtil; +import vip.xiaonuo.sys.modular.user.entity.SysUserPassword; +import vip.xiaonuo.sys.modular.user.mapper.SysUserPasswordMapper; +import vip.xiaonuo.sys.modular.user.service.SysUserPasswordService; + +import java.util.List; +import java.util.stream.Collectors; + +/** + * 用户密码Service接口实现类 + * + * @author yubaoshan + * @date 2024/12/21 01:25 + **/ +@Service +public class SysUserPasswordServiceImpl extends ServiceImpl implements SysUserPasswordService { + + @Override + public void insertUserPasswordHistory(String userId, String newPassword) { + SysUserPassword sysUserPassword = new SysUserPassword(); + sysUserPassword.setUserId(userId); + sysUserPassword.setPassword(CommonCryptogramUtil.doHashValue(newPassword)); + this.save(sysUserPassword); + } + + @Override + public List getUserPasswordHistoryLimit(String userId, int limitValue) { + return this.page(new Page<>(1, limitValue), new LambdaQueryWrapper() + .eq(SysUserPassword::getUserId, userId).orderByDesc(SysUserPassword::getCreateTime)) + .getRecords().stream().map(SysUserPassword::getPassword).collect(Collectors.toList()); + } +} diff --git a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserServiceImpl.java b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserServiceImpl.java index a2817d95..1c030601 100644 --- a/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserServiceImpl.java +++ b/snowy-plugin/snowy-plugin-sys/src/main/java/vip/xiaonuo/sys/modular/user/service/impl/SysUserServiceImpl.java @@ -79,6 +79,8 @@ import vip.xiaonuo.mobile.api.MobileButtonApi; import vip.xiaonuo.mobile.api.MobileMenuApi; import vip.xiaonuo.sys.core.enums.SysBuildInEnum; import vip.xiaonuo.sys.core.enums.SysDataTypeEnum; +import vip.xiaonuo.sys.core.util.SysEmailFormatUtl; +import vip.xiaonuo.sys.core.util.SysPasswordUtl; import vip.xiaonuo.sys.modular.group.entity.SysGroup; import vip.xiaonuo.sys.modular.group.service.SysGroupService; import vip.xiaonuo.sys.modular.org.entity.SysOrg; @@ -98,12 +100,18 @@ import vip.xiaonuo.sys.modular.resource.service.SysMenuService; import vip.xiaonuo.sys.modular.resource.service.SysModuleService; import vip.xiaonuo.sys.modular.role.entity.SysRole; import vip.xiaonuo.sys.modular.role.enums.SysRoleDataScopeCategoryEnum; +import vip.xiaonuo.sys.modular.role.param.SysRoleGrantUserParam; import vip.xiaonuo.sys.modular.role.service.SysRoleService; import vip.xiaonuo.sys.modular.user.entity.SysUser; +import vip.xiaonuo.sys.modular.user.entity.SysUserExt; +import vip.xiaonuo.sys.modular.user.enums.SysUpdatePasswordValidTypeEnum; +import vip.xiaonuo.sys.modular.user.enums.SysUserSourceFromTypeEnum; import vip.xiaonuo.sys.modular.user.enums.SysUserStatusEnum; import vip.xiaonuo.sys.modular.user.mapper.SysUserMapper; import vip.xiaonuo.sys.modular.user.param.*; import vip.xiaonuo.sys.modular.user.result.*; +import vip.xiaonuo.sys.modular.user.service.SysUserExtService; +import vip.xiaonuo.sys.modular.user.service.SysUserPasswordService; import vip.xiaonuo.sys.modular.user.service.SysUserService; import java.awt.image.BufferedImage; @@ -123,14 +131,72 @@ import java.util.stream.Collectors; @Service public class SysUserServiceImpl extends ServiceImpl implements SysUserService { - private static final String SNOWY_SYS_DEFAULT_PASSWORD_KEY = "SNOWY_SYS_DEFAULT_PASSWORD"; + /** B端验证码失效时间(适用图片验证码和短信验证码,单位:分钟,默认5分钟有效) */ + private static final String SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_B_KEY = "SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_B"; + /** B端重置密码验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B"; + + /** B端重置密码验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B"; + + /** B端修改密码验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B"; + + /** B端修改密码验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B"; + + /** B端重置密码成功短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B"; + + /** B端重置密码成功邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B"; + + /** B端密码即将到期短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B"; + + /** B端密码即将到期邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B"; + + /** B端绑定手机验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_B"; + + /** B端绑定邮箱验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_B"; + + /** B端修改绑定手机验证码短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_B"; + + /** B端修改绑定邮箱验证码邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_B"; + + /** B端注册账号成功短信消息模板 */ + private static final String SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B_KEY = "SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B"; + + /** B端注册账号成功邮件消息模板 */ + private static final String SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B_KEY = "SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B"; + + /** B端注册后是否需要绑定手机号 */ + private static final String SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_B_KEY = "SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_B"; + + /** B端注册后是否需要绑定邮箱 */ + private static final String SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_B_KEY = "SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_B"; + + /** B端新用户默认机构 */ + private static final String SNOWY_SYS_DEFAULT_NEW_USER_ORG_FOR_B_KEY = "SNOWY_SYS_DEFAULT_NEW_USER_ORG_FOR_B"; + + /** B端新用户默认职位 */ + private static final String SNOWY_SYS_DEFAULT_NEW_USER_POSITION_FOR_B_KEY = "SNOWY_SYS_DEFAULT_NEW_USER_POSITION_FOR_B"; + + /** B端新用户默认角色 */ + private static final String SNOWY_SYS_DEFAULT_NEW_USER_ROLE_FOR_B_KEY = "SNOWY_SYS_DEFAULT_NEW_USER_ROLE_FOR_B"; + + /** 工作台默认快捷方式 */ private static final String SNOWY_SYS_DEFAULT_WORKBENCH_DATA_KEY = "SNOWY_SYS_DEFAULT_WORKBENCH_DATA"; + /** 验证码缓存前缀 */ private static final String USER_VALID_CODE_CACHE_KEY = "user-validCode:"; - public static final String USER_CACHE_ALL_KEY = "sys-user:all"; - @Resource private CommonCacheOperator commonCacheOperator; @@ -179,6 +245,12 @@ public class SysUserServiceImpl extends ServiceImpl impl @Resource private SysGroupService sysGroupService; + @Resource + private SysUserExtService sysUserExtService; + + @Resource + private SysUserPasswordService sysUserPasswordService; + @Override public SysLoginUser getUserById(String id) { SysUser sysUser = this.getById(id); @@ -244,19 +316,26 @@ public class SysUserServiceImpl extends ServiceImpl impl @Transactional(rollbackFor = Exception.class) @Override - public void add(SysUserAddParam sysUserAddParam) { + public void add(SysUserAddParam sysUserAddParam, String sourceFromType) { checkParam(sysUserAddParam); SysUser sysUser = BeanUtil.toBean(sysUserAddParam, SysUser.class); if (ObjectUtil.isEmpty(sysUser.getAvatar())) { // 设置默认头像 sysUser.setAvatar(CommonAvatarUtil.generateImg(sysUser.getName())); } - // 设置默认密码 - sysUser.setPassword(CommonCryptogramUtil.doHashValue(devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_KEY))); + if(ObjectUtil.isEmpty(sysUser.getPassword())) { + // 设置默认密码 + sysUser.setPassword(CommonCryptogramUtil.doHashValue(SysPasswordUtl.getDefaultPassword())); + } else { + // 设置传入的密码 + sysUser.setPassword(CommonCryptogramUtil.doHashValue(sysUser.getPassword())); + } // 设置状态 sysUser.setUserStatus(SysUserStatusEnum.ENABLE.getValue()); + // 保存用户 this.save(sysUser); - + // 插入扩展信息 + sysUserExtService.createExtInfo(sysUser.getId(), sourceFromType); // 发布增加事件 CommonDataChangeEventCenter.doAddWithData(SysDataTypeEnum.USER.getValue(), JSONUtil.createArray().put(sysUser)); } @@ -276,7 +355,7 @@ public class SysUserServiceImpl extends ServiceImpl impl } } if (ObjectUtil.isNotEmpty(sysUserAddParam.getEmail())) { - if (!CommonEmailUtil.isEmail(sysUserAddParam.getEmail())) { + if (CommonEmailUtil.isNotEmail(sysUserAddParam.getEmail())) { throw new CommonException("邮箱:{}格式错误", sysUserAddParam.getEmail()); } if (this.count(new LambdaQueryWrapper() @@ -297,8 +376,8 @@ public class SysUserServiceImpl extends ServiceImpl impl throw new CommonException("不可修改系统内置超管用户账号"); } BeanUtil.copyProperties(sysUserEditParam, sysUser); + // 更新用户 this.updateById(sysUser); - // 发布更新事件 CommonDataChangeEventCenter.doUpdateWithData(SysDataTypeEnum.USER.getValue(), JSONUtil.createArray().put(sysUser)); } @@ -320,7 +399,7 @@ public class SysUserServiceImpl extends ServiceImpl impl } } if (ObjectUtil.isNotEmpty(sysUserEditParam.getEmail())) { - if (!CommonEmailUtil.isEmail(sysUserEditParam.getEmail())) { + if (CommonEmailUtil.isNotEmail(sysUserEditParam.getEmail())) { throw new CommonException("邮箱:{}格式错误", sysUserEditParam.getEmail()); } if (this.count(new LambdaQueryWrapper() @@ -364,6 +443,9 @@ public class SysUserServiceImpl extends ServiceImpl impl // 执行删除 this.removeByIds(sysUserIdList); + // 删除扩展信息 + sysUserExtService.remove(new LambdaQueryWrapper().in(SysUserExt::getUserId, sysUserIdList)); + // 发布删除事件 CommonDataChangeEventCenter.doDeleteWithDataId(SysDataTypeEnum.USER.getValue(), sysUserIdList); } @@ -382,16 +464,69 @@ public class SysUserServiceImpl extends ServiceImpl impl @Override public void enableUser(SysUserIdParam sysUserIdParam) { - this.update(new LambdaUpdateWrapper().eq(SysUser::getId, sysUserIdParam.getId()).set(SysUser::getUserStatus, SysUserStatusEnum.ENABLE.getValue())); } @Override public void resetPassword(SysUserIdParam sysUserIdParam) { + // 获取用户 + SysUser sysUser = this.queryEntity(sysUserIdParam.getId()); + // 获取默认密码 + String defaultPassword = SysPasswordUtl.getDefaultPassword(); + // 修改密码 this.update(new LambdaUpdateWrapper().eq(SysUser::getId, sysUserIdParam.getId()).set(SysUser::getPassword, - CommonCryptogramUtil.doHashValue(devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_KEY)))); + CommonCryptogramUtil.doHashValue(defaultPassword))); + // 获取手机号 + String phone = sysUser.getPhone(); + // 手机号不为空则发送密码重置成功短信 + if(ObjectUtil.isNotEmpty(phone)){ + // 重置密码成功短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("userNewPassword", defaultPassword); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 编码不为空 + if(ObjectUtil.isNotEmpty(codeValue)){ + try { + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } catch (Exception e) { + log.error(">>> 短信发送失败", e); + } + } + } + } + // 获取手机号 + String email = sysUser.getEmail(); + // 密码不为空则发送密码重置成功邮件 + if(ObjectUtil.isNotEmpty(email)){ + // 重置密码成功邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("userNewPassword", defaultPassword); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap); + try { + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } catch (Exception e) { + log.error(">>> 邮件发送失败", e); + } + } + } } @Override @@ -410,8 +545,10 @@ public class SysUserServiceImpl extends ServiceImpl impl sysUserPicValidCodeResult.setValidCodeBase64(validCodeBase64); // 将请求号返回前端 sysUserPicValidCodeResult.setValidCodeReqNo(validCodeReqNo); - // 将请求号作为key,验证码的值作为value放到redis,用于校验,5分钟有效 - commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + validCodeReqNo, validCode, 5 * 60); + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + validCodeReqNo, validCode, validCodeExpiredDuration); return sysUserPicValidCodeResult; } @@ -447,9 +584,19 @@ public class SysUserServiceImpl extends ServiceImpl impl @Override public String findPasswordGetPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { + return this.getPhoneValidCode(sysUserGetPhoneValidCodeParam, "重置密码验证码短信消息模板编码", SNOWY_SMS_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B_KEY); + } + + /** + * 获取手机验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private String getPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam, String templateCodeName, String templateCodeValue) { // 手机号 String phone = sysUserGetPhoneValidCodeParam.getPhone(); - // 验证码正确则校验手机号格式 + // 校验手机号格式 if (!PhoneUtil.isMobile(phone)) { throw new CommonException("手机号码:{}格式错误", phone); } @@ -457,85 +604,484 @@ public class SysUserServiceImpl extends ServiceImpl impl validValidCode(null, sysUserGetPhoneValidCodeParam.getValidCode(), sysUserGetPhoneValidCodeParam.getValidCodeReqNo()); // 根据手机号获取用户信息,判断用户是否存在 if (ObjectUtil.isEmpty(this.getUserByPhone(phone))) { - throw new CommonException("手机码:{}不存在", phone); + throw new CommonException("手机号码:{}不存在对应用户", phone); } // 生成手机验证码的值,随机6为数字 String phoneValidCode = RandomUtil.randomNumbers(6); // 生成手机验证码的请求号 String phoneValidCodeReqNo = IdWorker.getIdStr(); - - // TODO 使用阿里云执行发送验证码,将验证码作为短信内容的参数变量放入, - // TODO 签名不传则使用系统默认配置的签名,支持传入多个参数,示例:{"name":"张三","number":"15038****76"} - //devSmsApi.sendSmsAliyun(phone, null, "验证码模板号", JSONUtil.toJsonStr(JSONUtil.createObj().set("validCode", phoneValidCode))); - - // TODO 使用腾讯云执行发送验证码,将验证码作为短信内容的参数变量放入, - // TODO sdkAppId和签名不传则使用系统默认配置的sdkAppId和签名,支持传入多个参数,逗号拼接,示例:"张三,15038****76,进行中" - //devSmsApi.sendSmsTencent(null, phone, null, "验证码模板号", phoneValidCode); - - // 将请求号作为key,验证码的值作为value放到redis,用于校验,5分钟有效 - commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, 5 * 60); + // 短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(templateCodeValue); + if(ObjectUtil.isEmpty(smsTemplateCode)){ + throw new CommonException("请联系管理员配置{}", templateCodeName); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("validCode", phoneValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, validCodeExpiredDuration); // 返回请求号 return phoneValidCodeReqNo; } @Override public String findPasswordGetEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { + return this.getEmailValidCode(sysUserGetEmailValidCodeParam, "重置密码验证码邮件消息模板内容", SNOWY_EMAIL_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B_KEY); + } + + /** + * 获取邮箱验证码 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private String getEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam, String templateContentName, String templateContentValue) { // 邮箱 String email = sysUserGetEmailValidCodeParam.getEmail(); - // 验证码正确则校验邮箱格式 - if (!CommonEmailUtil.isEmail(email)) { + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { throw new CommonException("邮箱:{}格式错误", email); } // 执行校验验证码 validValidCode(null, sysUserGetEmailValidCodeParam.getValidCode(), sysUserGetEmailValidCodeParam.getValidCodeReqNo()); // 根据邮箱获取用户信息,判断用户是否存在 if (ObjectUtil.isEmpty(this.getUserByEmail(email))) { - throw new CommonException("邮箱:{}不存在", email); + throw new CommonException("邮箱:{}不存在对应用户", email); } // 生成邮箱验证码的值,随机6为数字 String emailValidCode = RandomUtil.randomNumbers(6); // 生成邮箱验证码的请求号 String emailValidCodeReqNo = IdWorker.getIdStr(); - - // TODO 使用本地发送邮件 - String content = "您正在找回密码,验证码为:" + emailValidCode + ",5分钟内有效。"; - devEmailApi.sendTextEmailLocal(email, "找回密码邮件", content, CollectionUtil.newArrayList()); - - // 将请求号作为key,验证码的值作为value放到redis,用于校验,5分钟有效 - commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + email + StrUtil.UNDERLINE + emailValidCodeReqNo, emailValidCode, 5 * 60); + // 邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(templateContentValue); + if(ObjectUtil.isEmpty(emailTemplateContent)){ + throw new CommonException("请联系管理员配置{}", templateContentName); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("validCode", emailValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + email + StrUtil.UNDERLINE + emailValidCodeReqNo, emailValidCode, validCodeExpiredDuration); // 返回请求号 return emailValidCodeReqNo; } @Override public void findPasswordByPhone(SysUserFindPwdByPhoneParam sysUserFindPwdByPhoneParam) { + // 再次校验手机号是否合法 + String phone = sysUserFindPwdByPhoneParam.getPhone(); + // 校验手机号格式 + if (!PhoneUtil.isMobile(phone)) { + throw new CommonException("手机号码:{}格式错误", phone); + } + // 根据手机号获取用户信息,判断用户是否存在 + SysLoginUser sysLoginUser = this.getUserByPhone(phone); + if (ObjectUtil.isEmpty(sysLoginUser)) { + throw new CommonException("手机号码:{}不存在对应用户", phone); + } // 执行校验验证码 - validValidCode(sysUserFindPwdByPhoneParam.getPhone(), sysUserFindPwdByPhoneParam.getValidCode(), sysUserFindPwdByPhoneParam.getValidCodeReqNo()); + validValidCode(phone, sysUserFindPwdByPhoneParam.getValidCode(), sysUserFindPwdByPhoneParam.getValidCodeReqNo()); + // 获取新密码 + String newPassword = CommonCryptogramUtil.doSm2Decrypt(sysUserFindPwdByPhoneParam.getNewPassword()).trim(); + // 校验新密码 + SysPasswordUtl.validNewPassword(sysLoginUser, newPassword); + // 修改密码 this.update(new LambdaUpdateWrapper().eq(SysUser::getPhone, - CommonCryptogramUtil.doSm4CbcEncrypt(sysUserFindPwdByPhoneParam.getPhone())).set(SysUser::getPassword, - CommonCryptogramUtil.doHashValue(CommonCryptogramUtil.doSm2Decrypt(sysUserFindPwdByPhoneParam.getNewPassword())))); + CommonCryptogramUtil.doSm4CbcEncrypt(phone)).set(SysUser::getPassword, + CommonCryptogramUtil.doHashValue(newPassword))); + // 更新用户最新修改密码时间 + sysUserExtService.updatePasswordLastTime(sysLoginUser.getId()); + // 追加用户历史密码信息 + sysUserPasswordService.insertUserPasswordHistory(sysLoginUser.getId(), newPassword); + // 重置密码成功短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("userNewPassword", newPassword); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 编码不为空 + if(ObjectUtil.isNotEmpty(codeValue)){ + try { + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } catch (Exception e) { + log.error(">>> 短信发送失败", e); + } + } + } } @Override public void findPasswordByEmail(SysUserFindPwdByEmailParam sysUserFindPwdByEmailParam) { + // 再次校验邮箱是否合法 + String email = sysUserFindPwdByEmailParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 根据邮箱获取用户信息,判断用户是否存在 + SysLoginUser sysLoginUser = this.getUserByEmail(email); + if (ObjectUtil.isEmpty(sysLoginUser)) { + throw new CommonException("邮箱:{}不存在对应用户", email); + } // 执行校验验证码 - validValidCode(sysUserFindPwdByEmailParam.getEmail(), sysUserFindPwdByEmailParam.getValidCode(), sysUserFindPwdByEmailParam.getValidCodeReqNo()); - this.update(new LambdaUpdateWrapper().eq(SysUser::getEmail, - sysUserFindPwdByEmailParam.getEmail()).set(SysUser::getPassword, - CommonCryptogramUtil.doHashValue(CommonCryptogramUtil.doSm2Decrypt(sysUserFindPwdByEmailParam.getNewPassword())))); + validValidCode(email, sysUserFindPwdByEmailParam.getValidCode(), sysUserFindPwdByEmailParam.getValidCodeReqNo()); + // 获取新密码 + String newPassword = CommonCryptogramUtil.doSm2Decrypt(sysUserFindPwdByEmailParam.getNewPassword()).trim(); + // 更新用户最新修改密码时间 + sysUserExtService.updatePasswordLastTime(sysLoginUser.getId()); + // 追加用户历史密码信息 + sysUserPasswordService.insertUserPasswordHistory(sysLoginUser.getId(), newPassword); + SysPasswordUtl.validNewPassword(sysLoginUser, newPassword); + // 修改密码 + this.update(new LambdaUpdateWrapper().eq(SysUser::getEmail, email).set(SysUser::getPassword, + CommonCryptogramUtil.doHashValue(newPassword))); + // 重置密码成功邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("userNewPassword", newPassword); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + try { + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } catch (Exception e) { + log.error(">>> 邮件发送失败", e); + } + } } @Override - public void updatePassword(SysUserUpdatePwdParam sysUserUpdatePwdParam) { + public String updatePasswordGetPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { + // 判断密码验证方式 + SysPasswordUtl.validUpdatePasswordValidType(SysUpdatePasswordValidTypeEnum.PHONE.getValue()); + return this.getPhoneValidCode(sysUserGetPhoneValidCodeParam, "修改密码验证码短信消息模板编码", SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B_KEY); + } + + @Override + public String updatePasswordGetEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { + // 判断密码验证方式 + SysPasswordUtl.validUpdatePasswordValidType(SysUpdatePasswordValidTypeEnum.EMAIL.getValue()); + return this.getEmailValidCode(sysUserGetEmailValidCodeParam, "修改密码验证码邮件消息模板内容", SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B_KEY); + } + + @Override + public void updatePasswordByOld(SysUserUpdatePwdByOldParam sysUserUpdatePwdByOldParam) { + // 判断密码验证方式 + SysPasswordUtl.validUpdatePasswordValidType(SysUpdatePasswordValidTypeEnum.OLD.getValue()); SysUser sysUser = this.queryEntity(StpUtil.getLoginIdAsString()); - String password = sysUserUpdatePwdParam.getPassword(); - String newPassword = sysUserUpdatePwdParam.getNewPassword(); + String password = CommonCryptogramUtil.doSm2Decrypt(sysUserUpdatePwdByOldParam.getPassword()).trim(); + String newPassword = CommonCryptogramUtil.doSm2Decrypt(sysUserUpdatePwdByOldParam.getNewPassword()).trim(); if (!CommonCryptogramUtil.doHashValue(password).equals(sysUser.getPassword())) { throw new CommonException("原密码错误"); } + // 校验新密码 + SysPasswordUtl.validNewPassword(sysUser, newPassword); + // 修改密码 this.update(new LambdaUpdateWrapper().eq(SysUser::getId, sysUser.getId()).set(SysUser::getPassword, CommonCryptogramUtil.doHashValue(newPassword))); + // 更新用户最新修改密码时间 + sysUserExtService.updatePasswordLastTime(sysUser.getId()); + // 追加用户历史密码信息 + sysUserPasswordService.insertUserPasswordHistory(sysUser.getId(), newPassword); + // 获取手机号 + String phone = sysUser.getPhone(); + // 手机号不为空则发送密码重置成功短信 + if(ObjectUtil.isNotEmpty(phone)){ + // 重置密码成功短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("userNewPassword", newPassword); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 编码不为空 + if(ObjectUtil.isNotEmpty(codeValue)){ + try { + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } catch (Exception e) { + log.error(">>> 短信发送失败", e); + } + } + } + } + // 获取手机号 + String email = sysUser.getEmail(); + // 密码不为空则发送密码重置成功邮件 + if(ObjectUtil.isNotEmpty(email)){ + // 重置密码成功邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("userNewPassword", newPassword); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + try { + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } catch (Exception e) { + log.error(">>> 邮件发送失败", e); + } + } + } + } + + @Override + public void updatePasswordByPhone(SysUserUpdatePwdByPhoneParam sysUserUpdatePwdByPhoneParam) { + // 判断密码验证方式 + SysPasswordUtl.validUpdatePasswordValidType(SysUpdatePasswordValidTypeEnum.PHONE.getValue()); + SysUserFindPwdByPhoneParam sysUserFindPwdByPhoneParam = new SysUserFindPwdByPhoneParam(); + BeanUtil.copyProperties(sysUserUpdatePwdByPhoneParam, sysUserFindPwdByPhoneParam); + this.findPasswordByPhone(sysUserFindPwdByPhoneParam); + } + + @Override + public void updatePasswordByEmail(SysUserUpdatePwdByEmailParam sysUserUpdatePwdByEmailParam) { + // 判断密码验证方式 + SysPasswordUtl.validUpdatePasswordValidType(SysUpdatePasswordValidTypeEnum.EMAIL.getValue()); + SysUserFindPwdByEmailParam sysUserFindPwdByEmailParam = new SysUserFindPwdByEmailParam(); + BeanUtil.copyProperties(sysUserUpdatePwdByEmailParam, sysUserFindPwdByEmailParam); + this.findPasswordByEmail(sysUserFindPwdByEmailParam); + } + + @Override + public String bindPhoneGetPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { + // 手机号 + String phone = sysUserGetPhoneValidCodeParam.getPhone(); + // 校验手机号格式 + if (!PhoneUtil.isMobile(phone)) { + throw new CommonException("手机号码:{}格式错误", phone); + } + // 执行校验验证码 + validValidCode(null, sysUserGetPhoneValidCodeParam.getValidCode(), sysUserGetPhoneValidCodeParam.getValidCodeReqNo()); + // 根据手机号获取用户信息,判断用户是否存在,如果存在则不能绑定该手机号 + if (ObjectUtil.isNotEmpty(this.getUserByPhone(phone))) { + throw new CommonException("手机号码:{}已存在对应用户", phone); + } + // 生成手机验证码的值,随机6为数字 + String phoneValidCode = RandomUtil.randomNumbers(6); + // 生成手机验证码的请求号 + String phoneValidCodeReqNo = IdWorker.getIdStr(); + // 绑定手机验证码短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_B_KEY); + if(ObjectUtil.isEmpty(smsTemplateCode)){ + throw new CommonException("请联系管理员配置绑定手机验证码短信消息模板编码"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("validCode", phoneValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, validCodeExpiredDuration); + // 返回请求号 + return phoneValidCodeReqNo; + } + + @Override + public String updateBindPhoneGetPhoneValidCode(SysUserGetPhoneValidCodeParam sysUserGetPhoneValidCodeParam) { + // 手机号 + String phone = sysUserGetPhoneValidCodeParam.getPhone(); + // 校验手机号格式 + if (!PhoneUtil.isMobile(phone)) { + throw new CommonException("手机号码:{}格式错误", phone); + } + // 执行校验验证码 + validValidCode(null, sysUserGetPhoneValidCodeParam.getValidCode(), sysUserGetPhoneValidCodeParam.getValidCodeReqNo()); + // 根据手机号获取用户信息,判断用户是否存在,如果存在则不能绑定该手机号 + if (ObjectUtil.isNotEmpty(this.getUserByPhone(phone))) { + throw new CommonException("手机号码:{}已存在对应用户", phone); + } + // 生成手机验证码的值,随机6为数字 + String phoneValidCode = RandomUtil.randomNumbers(6); + // 生成手机验证码的请求号 + String phoneValidCodeReqNo = IdWorker.getIdStr(); + // 修改绑定手机验证码短信消息模板编码 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_B_KEY); + if(ObjectUtil.isEmpty(smsTemplateCode)){ + throw new CommonException("请联系管理员配置修改绑定手机验证码短信消息模板编码"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone).set("validCode", phoneValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + phone + StrUtil.UNDERLINE + phoneValidCodeReqNo, phoneValidCode, validCodeExpiredDuration); + // 返回请求号 + return phoneValidCodeReqNo; + } + + @Override + public void bindPhone(SysUserBindPhoneParam sysUserBindPhoneParam) { + // 再次校验手机号是否合法 + String phone = sysUserBindPhoneParam.getPhone(); + // 校验手机号格式 + if (!PhoneUtil.isMobile(phone)) { + throw new CommonException("手机号码:{}格式错误", phone); + } + // 根据手机号获取用户信息,判断用户是否存在,如果存在则不能绑定该手机号 + if (ObjectUtil.isNotEmpty(this.getUserByPhone(phone))) { + throw new CommonException("手机号码:{}已存在对应用户", phone); + } + // 执行校验验证码 + validValidCode(phone, sysUserBindPhoneParam.getValidCode(), sysUserBindPhoneParam.getValidCodeReqNo()); + // 修改手机号 + this.update(new LambdaUpdateWrapper().eq(SysUser::getId, StpUtil.getLoginIdAsString()) + .set(SysUser::getPhone, CommonCryptogramUtil.doSm4CbcEncrypt(phone))); + } + + @Override + public String bindEmailGetEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { + // 邮箱 + String email = sysUserGetEmailValidCodeParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 执行校验验证码 + validValidCode(null, sysUserGetEmailValidCodeParam.getValidCode(), sysUserGetEmailValidCodeParam.getValidCodeReqNo()); + // 根据邮箱获取用户信息,判断用户是否存在,如果存在则不能绑定该邮箱 + if (ObjectUtil.isEmpty(this.getUserByEmail(email))) { + throw new CommonException("邮箱:{}已存在对应用户", email); + } + // 生成邮箱验证码的值,随机6为数字 + String emailValidCode = RandomUtil.randomNumbers(6); + // 生成邮箱验证码的请求号 + String emailValidCodeReqNo = IdWorker.getIdStr(); + // 绑定邮箱验证码邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_B_KEY); + if(ObjectUtil.isEmpty(emailTemplateContent)){ + throw new CommonException("请联系管理员配置绑定邮箱验证码邮件消息模板内容"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("validCode", emailValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + email + StrUtil.UNDERLINE + emailValidCodeReqNo, emailValidCode, validCodeExpiredDuration); + // 返回请求号 + return emailValidCodeReqNo; + } + + @Override + public String updateBindEmailGetEmailValidCode(SysUserGetEmailValidCodeParam sysUserGetEmailValidCodeParam) { + // 邮箱 + String email = sysUserGetEmailValidCodeParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 执行校验验证码 + validValidCode(null, sysUserGetEmailValidCodeParam.getValidCode(), sysUserGetEmailValidCodeParam.getValidCodeReqNo()); + // 根据邮箱获取用户信息,判断用户是否存在,如果存在则不能绑定该邮箱 + if (ObjectUtil.isEmpty(this.getUserByEmail(email))) { + throw new CommonException("邮箱:{}已存在对应用户", email); + } + // 生成邮箱验证码的值,随机6为数字 + String emailValidCode = RandomUtil.randomNumbers(6); + // 生成邮箱验证码的请求号 + String emailValidCodeReqNo = IdWorker.getIdStr(); + // 修改绑定邮箱验证码邮件消息模板内容 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_B_KEY); + if(ObjectUtil.isEmpty(emailTemplateContent)){ + throw new CommonException("请联系管理员配置修改绑定邮箱验证码邮件消息模板内容"); + } + // 获取验证码失效时间(单位:秒) + long validCodeExpiredDuration = this.getValidCodeExpiredDuration(); + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email).set("validCode", emailValidCode) + .set("validTime", validCodeExpiredDuration/60); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + // 将请求号作为key,验证码的值作为value放到redis,用于校验 + commonCacheOperator.put(USER_VALID_CODE_CACHE_KEY + email + StrUtil.UNDERLINE + emailValidCodeReqNo, emailValidCode, validCodeExpiredDuration); + // 返回请求号 + return emailValidCodeReqNo; + } + + @Override + public void bindEmail(SysUserBindEmailParam sysUserBindEmailParam) { + // 再次校验邮箱是否合法 + String email = sysUserBindEmailParam.getEmail(); + // 校验邮箱格式 + if (CommonEmailUtil.isNotEmail(email)) { + throw new CommonException("邮箱:{}格式错误", email); + } + // 根据邮箱获取用户信息,判断用户是否存在,如果存在则不能绑定该邮箱 + if (ObjectUtil.isEmpty(this.getUserByEmail(email))) { + throw new CommonException("邮箱:{}已存在对应用户", email); + } + // 执行校验验证码 + validValidCode(email, sysUserBindEmailParam.getValidCode(), sysUserBindEmailParam.getValidCodeReqNo()); + // 修改邮箱 + this.update(new LambdaUpdateWrapper().eq(SysUser::getId, StpUtil.getLoginIdAsString()) + .set(SysUser::getEmail, email)); } @Override @@ -629,7 +1175,7 @@ public class SysUserServiceImpl extends ServiceImpl impl // 获取拥有的菜单列表 List menuList = allMenuList.stream().filter(sysMenu -> - menuIdList.contains(sysMenu.getId())).collect(Collectors.toList()); + menuIdList.contains(sysMenu.getId())).toList(); // 对获取到的角色对应的菜单列表进行处理,获取父列表 menuList.forEach(sysMenu -> execRecursionFindParent(allMenuList, sysMenu.getId(), resultList)); @@ -703,7 +1249,7 @@ public class SysUserServiceImpl extends ServiceImpl impl if (!sysMenu.getMenuType().equals(SysResourceMenuTypeEnum.CATALOG.getValue())) { metaJsonObject.set("type", sysMenu.getMenuType().toLowerCase()); } - if (sysMenu.getId().equals(sysMenus.orElse(null).getId())) { + if (sysMenus.orElse(null) != null && sysMenu.getId().equals(sysMenus.orElse(null).getId())) { // 如果是首页,则设置affix metaJsonObject.set("affix", true); } @@ -714,12 +1260,12 @@ public class SysUserServiceImpl extends ServiceImpl impl } menuJsonObject.set("meta", metaJsonObject); return menuJsonObject; - }).collect(Collectors.toList()); + }).toList(); // 执行构造树 List> treeNodeList = resultJsonObjectList.stream().map(jsonObject -> - new TreeNode<>(jsonObject.getStr("id"), jsonObject.getStr("parentId"), - jsonObject.getStr("title"), jsonObject.getInt("sortCode")).setExtra(JSONUtil.parseObj(jsonObject))) + new TreeNode<>(jsonObject.getStr("id"), jsonObject.getStr("parentId"), + jsonObject.getStr("title"), jsonObject.getInt("sortCode")).setExtra(JSONUtil.parseObj(jsonObject))) .collect(Collectors.toList()); return TreeUtil.build(treeNodeList, "0"); } @@ -801,10 +1347,10 @@ public class SysUserServiceImpl extends ServiceImpl impl } } } + List extJsonList = sysUserGrantResourceParam.getGrantInfoList().stream() + .map(JSONUtil::toJsonStr).collect(Collectors.toList()); + sysRelationService.saveRelationBatchWithClear(sysUserGrantResourceParam.getId(), menuIdList, SysRelationCategoryEnum.SYS_USER_HAS_RESOURCE.getValue(), extJsonList); } - List extJsonList = sysUserGrantResourceParam.getGrantInfoList().stream() - .map(JSONUtil::toJsonStr).collect(Collectors.toList()); - sysRelationService.saveRelationBatchWithClear(sysUserGrantResourceParam.getId(), menuIdList, SysRelationCategoryEnum.SYS_USER_HAS_RESOURCE.getValue(), extJsonList); } @Override @@ -832,8 +1378,13 @@ public class SysUserServiceImpl extends ServiceImpl impl public List> loginOrgTree(SysUserIdParam sysUserIdParam) { SysUser sysUser = this.queryEntity(sysUserIdParam.getId()); List originDataList = sysOrgService.getAllOrgList(); - List sysOrgList = sysOrgService.getParentListById(originDataList, sysUser.getOrgId(), true); - List> treeNodeList = sysOrgList.stream().map(sysOrg -> { + // 构建一个根组织 + SysOrg rootSysOrg = new SysOrg(); + rootSysOrg.setId("0"); + rootSysOrg.setParentId("-1"); + rootSysOrg.setName("根组织"); + originDataList.add(rootSysOrg); + List> treeNodeList = originDataList.stream().map(sysOrg -> { TreeNode treeNode = new TreeNode<>(sysOrg.getId(), sysOrg.getParentId(), sysOrg.getName(), sysOrg.getSortCode()); if (sysOrg.getId().equals(sysUser.getOrgId())) { treeNode.setExtra(JSONUtil.createObj().set("style", JSONUtil.createObj().set("color", "#FFF") @@ -841,35 +1392,21 @@ public class SysUserServiceImpl extends ServiceImpl impl } return treeNode; }).collect(Collectors.toList()); - return TreeUtil.build(treeNodeList, "0", new TreeNodeConfig().setParentIdKey("pid") + return TreeUtil.build(treeNodeList, "-1", new TreeNodeConfig().setParentIdKey("pid") .setNameKey("label"), new DefaultNodeParser<>()); } @Override public void updateUserInfo(SysUserUpdateInfoParam sysUserUpdateInfoParam) { String id = StpLoginUserUtil.getLoginUser().getId(); - if (!StrUtil.equals(id,sysUserUpdateInfoParam.getId())){ - throw new CommonException("禁止修改他人信息"); + if (!StrUtil.equals(id, sysUserUpdateInfoParam.getId())){ + throw new CommonException("被修改用户与当前登录用户不匹配"); } - SysUser sysUser = this.queryEntity(sysUserUpdateInfoParam.getId()); - - if (ObjectUtil.isNotEmpty(sysUserUpdateInfoParam.getPhone())) { - if (!PhoneUtil.isMobile(sysUserUpdateInfoParam.getPhone())) { - throw new CommonException("手机号码:{}格式错误", sysUserUpdateInfoParam.getPhone()); - } - if (this.count(new LambdaQueryWrapper().ne(SysUser::getId, sysUser.getId()) - .eq(SysUser::getPhone, CommonCryptogramUtil.doSm4CbcEncrypt(sysUserUpdateInfoParam.getPhone()))) > 0) { - throw new CommonException("存在重复的手机号,手机号为:{}", sysUserUpdateInfoParam.getPhone()); - } - } LambdaUpdateWrapper lambdaUpdateWrapper = new LambdaUpdateWrapper().eq(SysUser::getId, sysUser.getId()); if(ObjectUtil.isNotEmpty(sysUserUpdateInfoParam.getName())) { lambdaUpdateWrapper.set(SysUser::getName, sysUserUpdateInfoParam.getName()); } - if(ObjectUtil.isNotEmpty(sysUserUpdateInfoParam.getPhone())) { - lambdaUpdateWrapper.set(SysUser::getPhone, CommonCryptogramUtil.doSm4CbcEncrypt(sysUserUpdateInfoParam.getPhone())); - } if(ObjectUtil.isNotEmpty(sysUserUpdateInfoParam.getNickname())) { lambdaUpdateWrapper.set(SysUser::getNickname, sysUserUpdateInfoParam.getNickname()); } @@ -879,12 +1416,6 @@ public class SysUserServiceImpl extends ServiceImpl impl if(ObjectUtil.isNotEmpty(sysUserUpdateInfoParam.getBirthday())) { lambdaUpdateWrapper.set(SysUser::getBirthday, sysUserUpdateInfoParam.getBirthday()); } - if(ObjectUtil.isNotEmpty(sysUserUpdateInfoParam.getEmail())) { - lambdaUpdateWrapper.set(SysUser::getEmail, sysUserUpdateInfoParam.getEmail()); - } - if(ObjectUtil.isNotEmpty(sysUserUpdateInfoParam.getSignature())) { - lambdaUpdateWrapper.set(SysUser::getSignature, sysUserUpdateInfoParam.getSignature()); - } // 更新指定字段 this.update(lambdaUpdateWrapper); } @@ -951,7 +1482,7 @@ public class SysUserServiceImpl extends ServiceImpl impl } }); if (ObjectUtil.isNotEmpty(buttonIdList)) { - return mobileButtonApi.listByIds(buttonIdList); + return mobileButtonApi.listButtonCodeListByIdList(buttonIdList); } return CollectionUtil.newArrayList(); } @@ -969,10 +1500,10 @@ public class SysUserServiceImpl extends ServiceImpl impl public List getScopeListByMap(Map> groupMap, String orgId) { List resultList = CollectionUtil.newArrayList(); List sysOrgList = sysOrgService.getAllOrgList(); - List scopeAllList = sysOrgList.stream().map(SysOrg::getId).collect(Collectors.toList()); + List scopeAllList = sysOrgList.stream().map(SysOrg::getId).toList(); List scopeOrgList = CollectionUtil.newArrayList(orgId); List scopeOrgChildList = sysOrgService.getChildListById(sysOrgList, orgId, true) - .stream().map(SysOrg::getId).collect(Collectors.toList()); + .stream().map(SysOrg::getId).toList(); groupMap.forEach((key, value) -> { JSONObject jsonObject = JSONUtil.createObj().set("apiUrl", key); Set scopeSet = CollectionUtil.newHashSet(); @@ -1139,7 +1670,7 @@ public class SysUserServiceImpl extends ServiceImpl impl // 设置默认头像 sysUser.setAvatar(CommonAvatarUtil.generateImg(sysUser.getName())); // 设置默认密码 - sysUser.setPassword(CommonCryptogramUtil.doHashValue(devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_PASSWORD_KEY))); + sysUser.setPassword(CommonCryptogramUtil.doHashValue(SysPasswordUtl.getDefaultPassword())); // 设置排序码 sysUser.setSortCode(99); // 设置状态 @@ -1160,6 +1691,10 @@ public class SysUserServiceImpl extends ServiceImpl impl // 保存或更新 this.saveOrUpdate(sysUser); + // 如果是增加,则同时增加用户扩展信息 + if(isAdd) { + sysUserExtService.createExtInfo(sysUser.getId(), SysUserSourceFromTypeEnum.SYSTEM_ADD.getValue()); + } // 返回成功 return JSONUtil.createObj().set("success", true); } catch (Exception e) { @@ -1427,7 +1962,8 @@ public class SysUserServiceImpl extends ServiceImpl impl public Page getAllUserSelectorList() { return this.page(CommonPageRequest.defaultPage(), new LambdaQueryWrapper().select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, SysUser::getName, - SysUser::getSortCode, SysUser::getGender, SysUser::getEntryDate).orderByAsc(SysUser::getSortCode)); + SysUser::getSortCode, SysUser::getGender, SysUser::getEntryDate) + .eq(SysUser::getUserStatus, SysUserStatusEnum.ENABLE.getValue()).orderByAsc(SysUser::getSortCode)); } /* ====用户部分所需要用到的选择器==== */ @@ -1436,66 +1972,68 @@ public class SysUserServiceImpl extends ServiceImpl impl public List> orgTreeSelector() { List sysOrgList = sysOrgService.getAllOrgList(); List> treeNodeList = sysOrgList.stream().map(sysOrg -> - new TreeNode<>(sysOrg.getId(), sysOrg.getParentId(), sysOrg.getName(), sysOrg.getSortCode())) + new TreeNode<>(sysOrg.getId(), sysOrg.getParentId(), sysOrg.getName(), sysOrg.getSortCode())) .collect(Collectors.toList()); return TreeUtil.build(treeNodeList, "0"); } @Override public Page orgListSelector(SysUserSelectorOrgListParam sysUserSelectorOrgListParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 - lambdaQueryWrapper.select(SysOrg::getId, SysOrg::getParentId, SysOrg::getName, + queryWrapper.lambda().select(SysOrg::getId, SysOrg::getParentId, SysOrg::getName, SysOrg::getCategory, SysOrg::getSortCode); if (ObjectUtil.isNotEmpty(sysUserSelectorOrgListParam.getParentId())) { - lambdaQueryWrapper.eq(SysOrg::getParentId, sysUserSelectorOrgListParam.getParentId()); + queryWrapper.lambda().eq(SysOrg::getParentId, sysUserSelectorOrgListParam.getParentId()); } if (ObjectUtil.isNotEmpty(sysUserSelectorOrgListParam.getSearchKey())) { - lambdaQueryWrapper.like(SysOrg::getName, sysUserSelectorOrgListParam.getSearchKey()); + queryWrapper.lambda().like(SysOrg::getName, sysUserSelectorOrgListParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysOrg::getSortCode); - return sysOrgService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysOrg::getSortCode); + return sysOrgService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } @Override public Page positionSelector(SysUserSelectorPositionParam sysUserSelectorPositionParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); // 查询部分字段 - lambdaQueryWrapper.select(SysPosition::getId, SysPosition::getOrgId, SysPosition::getName, + queryWrapper.lambda().select(SysPosition::getId, SysPosition::getOrgId, SysPosition::getName, SysPosition::getCategory, SysPosition::getSortCode); if (ObjectUtil.isNotEmpty(sysUserSelectorPositionParam.getOrgId())) { - lambdaQueryWrapper.eq(SysPosition::getOrgId, sysUserSelectorPositionParam.getOrgId()); + queryWrapper.lambda().eq(SysPosition::getOrgId, sysUserSelectorPositionParam.getOrgId()); } if (ObjectUtil.isNotEmpty(sysUserSelectorPositionParam.getSearchKey())) { - lambdaQueryWrapper.like(SysPosition::getName, sysUserSelectorPositionParam.getSearchKey()); + queryWrapper.lambda().like(SysPosition::getName, sysUserSelectorPositionParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysPosition::getSortCode); - return sysPositionService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysPosition::getSortCode); + return sysPositionService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } @Override public Page roleSelector(SysUserSelectorRoleParam sysUserSelectorRoleParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); - lambdaQueryWrapper.select(SysRole::getId, SysRole::getOrgId, SysRole::getName, + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + queryWrapper.lambda().select(SysRole::getId, SysRole::getOrgId, SysRole::getName, SysRole::getCode, SysRole::getCategory, SysRole::getSortCode); if (ObjectUtil.isNotEmpty(sysUserSelectorRoleParam.getOrgId())) { - lambdaQueryWrapper.eq(SysRole::getOrgId, sysUserSelectorRoleParam.getOrgId()); + queryWrapper.lambda().eq(SysRole::getOrgId, sysUserSelectorRoleParam.getOrgId()); } if (ObjectUtil.isNotEmpty(sysUserSelectorRoleParam.getCategory())) { - lambdaQueryWrapper.eq(SysRole::getCategory, sysUserSelectorRoleParam.getCategory()); + queryWrapper.lambda().eq(SysRole::getCategory, sysUserSelectorRoleParam.getCategory()); } if (ObjectUtil.isNotEmpty(sysUserSelectorRoleParam.getSearchKey())) { - lambdaQueryWrapper.like(SysRole::getName, sysUserSelectorRoleParam.getSearchKey()); + queryWrapper.lambda().like(SysRole::getName, sysUserSelectorRoleParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysRole::getSortCode); - return sysRoleService.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysRole::getSortCode); + return sysRoleService.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } @Override public Page userSelector(SysUserSelectorUserParam sysUserSelectorUserParam) { - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + QueryWrapper queryWrapper = new QueryWrapper().checkSqlInjection(); + // 只查询状态为正常的 + queryWrapper.lambda().eq(SysUser::getUserStatus, SysUserStatusEnum.ENABLE.getValue()); // 只查询部分字段 - lambdaQueryWrapper.select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, + queryWrapper.lambda().select(SysUser::getId, SysUser::getAvatar, SysUser::getOrgId, SysUser::getPositionId, SysUser::getAccount, SysUser::getName, SysUser::getSortCode, SysUser::getGender, SysUser::getEntryDate); // 如果查询条件为空,则直接查询 if(ObjectUtil.isAllEmpty(sysUserSelectorUserParam.getOrgId(), sysUserSelectorUserParam.getSearchKey())) { @@ -1504,18 +2042,18 @@ public class SysUserServiceImpl extends ServiceImpl impl if (ObjectUtil.isNotEmpty(sysUserSelectorUserParam.getOrgId())) { // 如果组织id不为空,则查询该组织及其子组织下的所有人 List childOrgIdList = CollStreamUtil.toList(sysOrgService.getChildListById(sysOrgService - .getAllOrgList(), sysUserSelectorUserParam.getOrgId(), true), SysOrg::getId); + .getAllOrgList(), sysUserSelectorUserParam.getOrgId(), true), SysOrg::getId); if (ObjectUtil.isNotEmpty(childOrgIdList)) { - lambdaQueryWrapper.in(SysUser::getOrgId, childOrgIdList); + queryWrapper.lambda().in(SysUser::getOrgId, childOrgIdList); } else { return new Page<>(); } } if (ObjectUtil.isNotEmpty(sysUserSelectorUserParam.getSearchKey())) { - lambdaQueryWrapper.like(SysUser::getName, sysUserSelectorUserParam.getSearchKey()); + queryWrapper.lambda().like(SysUser::getName, sysUserSelectorUserParam.getSearchKey()); } - lambdaQueryWrapper.orderByAsc(SysUser::getSortCode); - return this.page(CommonPageRequest.defaultPage(), lambdaQueryWrapper); + queryWrapper.lambda().orderByAsc(SysUser::getSortCode); + return this.page(CommonPageRequest.defaultPage(), queryWrapper.lambda()); } } @@ -1570,18 +2108,6 @@ public class SysUserServiceImpl extends ServiceImpl impl return sysPositionService.list(lambdaQueryWrapper); } - @Override - public List getGroupListByIdList(SysUserGroupIdListParam sysUserGroupIdListParam) { - if (ObjectUtil.isEmpty(sysUserGroupIdListParam.getIdList())) { - return CollectionUtil.newArrayList(); - } - LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); - // 查询部分字段 - lambdaQueryWrapper.select(SysGroup::getId, SysGroup::getName, SysGroup::getRemark, SysGroup::getSortCode) - .in(SysGroup::getId, sysUserGroupIdListParam.getIdList()).orderByAsc(SysGroup::getSortCode); - return sysGroupService.list(lambdaQueryWrapper); - } - @Override public List getRoleListByIdList(SysUserIdListParam sysUserIdListParam) { if (ObjectUtil.isEmpty(sysUserIdListParam.getIdList())) { @@ -1595,8 +2121,300 @@ public class SysUserServiceImpl extends ServiceImpl impl return sysRoleService.list(lambdaQueryWrapper); } + @Override + public List getGroupListByIdList(SysUserGroupIdListParam sysUserGroupIdListParam) { + if (ObjectUtil.isEmpty(sysUserGroupIdListParam.getIdList())) { + return CollectionUtil.newArrayList(); + } + LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>(); + // 查询部分字段 + lambdaQueryWrapper.select(SysGroup::getId, SysGroup::getName, SysGroup::getRemark, SysGroup::getSortCode) + .in(SysGroup::getId, sysUserGroupIdListParam.getIdList()).orderByAsc(SysGroup::getSortCode); + return sysGroupService.list(lambdaQueryWrapper); + } + @Override public String getAvatarById(SysUserIdParam sysUserIdParam) { return this.detail(sysUserIdParam).getAvatar(); } + + @Transactional(rollbackFor = Exception.class) + @Override + public SysUser createUserWithPhone(String phone) { + SysUserAddParam sysUserAddParam = new SysUserAddParam(); + sysUserAddParam.setAccount(phone); + sysUserAddParam.setName(phone); + sysUserAddParam.setPhone(phone); + sysUserAddParam.setOrgId(this.getDefaultNewUserOrgId()); + sysUserAddParam.setPositionId(this.getDefaultNewUserPositionId()); + // 保存用户 + this.add(sysUserAddParam, SysUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue()); + // 获取用户信息 + SysUser sysUser = this.getOne(new LambdaQueryWrapper().eq(SysUser::getPhone, CommonCryptogramUtil.doSm4CbcEncrypt(phone))); + // 授权默认角色 + SysRoleGrantUserParam sysRoleGrantUserParam = new SysRoleGrantUserParam(); + sysRoleGrantUserParam.setId(this.getDefaultNewUserRoleId()); + sysRoleGrantUserParam.setGrantInfoList(CollectionUtil.newArrayList(sysUser.getId())); + sysRoleService.grantUser(sysRoleGrantUserParam); + // 发送注册成功短信 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } + // 返回用户 + return sysUser; + } + + @Transactional(rollbackFor = Exception.class) + @Override + public SysUser createUserWithEmail(String email) { + SysUserAddParam sysUserAddParam = new SysUserAddParam(); + sysUserAddParam.setAccount(email); + sysUserAddParam.setName(email); + sysUserAddParam.setEmail(email); + sysUserAddParam.setOrgId(this.getDefaultNewUserOrgId()); + sysUserAddParam.setPositionId(this.getDefaultNewUserPositionId()); + // 保存用户 + this.add(sysUserAddParam, SysUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue()); + // 获取用户信息 + SysUser sysUser = this.getOne(new LambdaQueryWrapper().eq(SysUser::getEmail, email)); + // 授权默认角色 + SysRoleGrantUserParam sysRoleGrantUserParam = new SysRoleGrantUserParam(); + sysRoleGrantUserParam.setId(this.getDefaultNewUserRoleId()); + sysRoleGrantUserParam.setGrantInfoList(CollectionUtil.newArrayList(sysUser.getId())); + sysRoleService.grantUser(sysRoleGrantUserParam); + // 发送注册成功邮件 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B_KEY); + // 不为空才发送 + if(ObjectUtil.isNotEmpty(emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } + // 返回用户 + return sysUser; + } + + @Transactional(rollbackFor = Exception.class) + @Override + public SysUser createUserWithAccount(String account, String password) { + SysUserAddParam sysUserAddParam = new SysUserAddParam(); + sysUserAddParam.setAccount(account); + sysUserAddParam.setName(account); + sysUserAddParam.setPassword(password); + sysUserAddParam.setOrgId(this.getDefaultNewUserOrgId()); + sysUserAddParam.setPositionId(this.getDefaultNewUserPositionId()); + // 保存用户 + this.add(sysUserAddParam, SysUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue()); + // 获取用户信息 + SysUser sysUser = this.getOne(new LambdaQueryWrapper().eq(SysUser::getAccount, account)); + // 授权默认角色 + SysRoleGrantUserParam sysRoleGrantUserParam = new SysRoleGrantUserParam(); + sysRoleGrantUserParam.setId(this.getDefaultNewUserRoleId()); + sysRoleGrantUserParam.setGrantInfoList(CollectionUtil.newArrayList(sysUser.getId())); + sysRoleService.grantUser(sysRoleGrantUserParam); + // 返回用户 + return sysUser; + } + + @Override + public Boolean isUserNeedBindPhone() { + // 获取当前用户 + SysUser sysUser = this.queryEntity(StpUtil.getLoginIdAsString()); + // 查询当前用户是否注册的 + SysUserExt sysUserExt = sysUserExtService.getOne(new LambdaQueryWrapper().eq(SysUserExt::getUserId, StpUtil.getLoginIdAsString()) + .eq(SysUserExt::getSourceFromType, SysUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue())); + // 不为空,则判断手机号是否为空 + if(ObjectUtil.isNotEmpty(sysUserExt)){ + // 手机号为空,判断系统注册后是否需要绑定手机号 + if(ObjectUtil.isEmpty(sysUser.getPhone())) { + String registerNeedBindPhone = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(registerNeedBindPhone)){ + return Convert.toBool(registerNeedBindPhone); + } else { + return false; + } + } else { + return false; + } + } else { + return false; + } + } + + @Override + public Boolean isUserNeedBindEmail() { + // 获取当前用户 + SysUser sysUser = this.queryEntity(StpUtil.getLoginIdAsString()); + // 查询当前用户是否注册的 + SysUserExt sysUserExt = sysUserExtService.getOne(new LambdaQueryWrapper().eq(SysUserExt::getUserId, StpUtil.getLoginIdAsString()) + .eq(SysUserExt::getSourceFromType, SysUserSourceFromTypeEnum.SYSTEM_REGISTER.getValue())); + // 不为空,则判断邮箱是否为空 + if(ObjectUtil.isNotEmpty(sysUserExt)){ + // 邮箱为空,判断系统注册后是否需要绑定邮箱 + if(ObjectUtil.isEmpty(sysUser.getEmail())) { + String registerNeedBindEmail = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_B_KEY); + if(ObjectUtil.isNotEmpty(registerNeedBindEmail)){ + return Convert.toBool(registerNeedBindEmail); + } else { + return false; + } + } else { + return false; + } + } else { + return false; + } + } + + /** + * 获取新用户默认机构配置 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private String getDefaultNewUserOrgId() { + String defaultNewUserOrgId = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_NEW_USER_ORG_FOR_B_KEY); + if(ObjectUtil.isEmpty(defaultNewUserOrgId)){ + throw new CommonException("请联系管理员配置新用户默认机构"); + } + return defaultNewUserOrgId; + } + + /** + * 获取新用户默认职位配置 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private String getDefaultNewUserPositionId() { + String defaultNewUserPositionId = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_NEW_USER_POSITION_FOR_B_KEY); + if(ObjectUtil.isEmpty(defaultNewUserPositionId)){ + throw new CommonException("请联系管理员配置新用户默认职位"); + } + return defaultNewUserPositionId; + } + + /** + * 获取新用户默认角色配置 + * + * @author xuyuxiang + * @date 2022/8/25 15:16 + **/ + private String getDefaultNewUserRoleId() { + String defaultNewUserRoleId = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_NEW_USER_ROLE_FOR_B_KEY); + if(ObjectUtil.isEmpty(defaultNewUserRoleId)){ + throw new CommonException("请联系管理员配置新用户默认角色"); + } + return defaultNewUserRoleId; + } + + @Override + public Boolean isUserPasswordExpired() { + return SysPasswordUtl.isUserPasswordExpired(StpUtil.getLoginIdAsString()); + } + + @Override + public void noticeUserPasswordAboutToExpired() { + // 密码即将到期短信消息模板 + String smsTemplateCode = devConfigApi.getValueByKey(SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B_KEY); + // 密码即将到期邮件消息模板 + String emailTemplateContent = devConfigApi.getValueByKey(SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B_KEY); + // 获取今日需要提醒密码到期的用户集合 + SysPasswordUtl.thisDayPasswordExpiredNeedNoticeUserIdList().forEach(sysUser -> { + // 获取手机号 + String phone = sysUser.getPhone(); + // 不为空才发送 + if(ObjectUtil.isAllNotEmpty(phone, smsTemplateCode)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(smsTemplateCode); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userPhone", phone); + // 获取编码 + String codeValue = contentJSONObject.getStr("code"); + // 编码不为空 + if(ObjectUtil.isNotEmpty(codeValue)){ + try { + // 发送短信 + devSmsApi.sendDynamicSms(phone, codeValue, paramMap); + } catch (Exception e) { + log.error(">>> 短信发送失败", e); + } + } + } + // 获取邮箱 + String email = sysUser.getEmail(); + // 不为空才发送 + if(ObjectUtil.isAllNotEmpty(email, emailTemplateContent)){ + // 模板内容转为JSONObject + JSONObject contentJSONObject = JSONUtil.parseObj(emailTemplateContent); + // 定义变量参数 + JSONObject paramMap = JSONUtil.createObj().set("userEmail", email); + // 获取格式化后的主题 + String subject = SysEmailFormatUtl.format(contentJSONObject.getStr("subject"), paramMap);; + // 获取格式化后的内容 + String content = SysEmailFormatUtl.format(contentJSONObject.getStr("content"), paramMap);; + try { + // 发送邮件 + devEmailApi.sendDynamicHtmlEmail(email, subject, content); + } catch (Exception e) { + log.error(">>> 邮件发送失败", e); + } + } + }); + } + + @Transactional(rollbackFor = Exception.class) + @Override + public void doRegister(String account, String password) { + // 校验账号 + SysLoginUser sysLoginUser = this.getUserByAccount(account); + if(ObjectUtil.isNotEmpty(sysLoginUser)) { + throw new CommonException("账号已存在"); + } + // 校验密码 + SysPasswordUtl.validNewPassword(password); + // 根据账号密码创建用户 + this.createUserWithAccount(account, password); + } + + @Override + public JSONObject getUpdatePasswordValidConfig() { + return SysPasswordUtl.getUpdatePasswordValidConfig(); + } + + /** + * 获取验证码失效时间(单位:秒) + * + * @author xuyuxiang + * @date 2025/3/21 20:25 + **/ + private long getValidCodeExpiredDuration() { + // 默认5分钟 + int defaultExpiredTime = 5; + // 获取配置验证码失效时间 + String configCaptchaExpiredDuration = devConfigApi.getValueByKey(SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_B_KEY); + // 判断是否为空 + if(ObjectUtil.isNotEmpty(configCaptchaExpiredDuration)){ + // 配置了则使用配置的失效时间 + defaultExpiredTime = Convert.toInt(configCaptchaExpiredDuration); + } + // 转为秒 + return defaultExpiredTime * 60L; + } } diff --git a/snowy-web-app/pom.xml b/snowy-web-app/pom.xml index 713a3f22..3f7aa18b 100644 --- a/snowy-web-app/pom.xml +++ b/snowy-web-app/pom.xml @@ -16,6 +16,7 @@ + junit @@ -37,9 +38,9 @@ - - com.mysql - mysql-connector-j + + com.mysql + mysql-connector-j @@ -52,8 +53,8 @@ - diff --git a/snowy-web-app/src/main/java/vip/xiaonuo/core/config/GlobalConfigure.java b/snowy-web-app/src/main/java/vip/xiaonuo/core/config/GlobalConfigure.java index 4199eef4..c7b8fada 100644 --- a/snowy-web-app/src/main/java/vip/xiaonuo/core/config/GlobalConfigure.java +++ b/snowy-web-app/src/main/java/vip/xiaonuo/core/config/GlobalConfigure.java @@ -147,9 +147,6 @@ public class GlobalConfigure implements WebMvcConfigurer { /* 系统字典树 */ "/dev/dict/tree", - /* 文件下载 */ - "/dev/file/download", - /* 用户个人中心相关 */ "/sys/userCenter/getPicCaptcha", "/sys/userCenter/findPasswordGetPhoneValidCode", @@ -576,29 +573,19 @@ public class GlobalConfigure implements WebMvcConfigurer { @Component public static class CustomMetaObjectHandler implements MetaObjectHandler { - /** - * 删除标志 - */ + /** 删除标志 */ private static final String DELETE_FLAG = "deleteFlag"; - /** - * 创建人 - */ + /** 创建人 */ private static final String CREATE_USER = "createUser"; - /** - * 创建时间 - */ + /** 创建时间 */ private static final String CREATE_TIME = "createTime"; - /** - * 更新人 - */ + /** 更新人 */ private static final String UPDATE_USER = "updateUser"; - /** - * 更新时间 - */ + /** 更新时间 */ private static final String UPDATE_TIME = "updateTime"; @Override @@ -609,24 +596,21 @@ public class GlobalConfigure implements WebMvcConfigurer { if (ObjectUtil.isNull(deleteFlag)) { setFieldValByName(DELETE_FLAG, EnumUtil.toString(CommonDeleteFlagEnum.NOT_DELETE), metaObject); } - } catch (ReflectionException ignored) { - } + } catch (ReflectionException ignored) { } try { //为空则设置createUser Object createUser = metaObject.getValue(CREATE_USER); if (ObjectUtil.isNull(createUser)) { setFieldValByName(CREATE_USER, this.getUserId(), metaObject); } - } catch (ReflectionException ignored) { - } + } catch (ReflectionException ignored) { } try { //为空则设置createTime Object createTime = metaObject.getValue(CREATE_TIME); if (ObjectUtil.isNull(createTime)) { setFieldValByName(CREATE_TIME, DateTime.now(), metaObject); } - } catch (ReflectionException ignored) { - } + } catch (ReflectionException ignored) { } } @Override diff --git a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorAttributesHandler.java b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorAttributesHandler.java index 0320a17a..8177cf4d 100644 --- a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorAttributesHandler.java +++ b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorAttributesHandler.java @@ -60,8 +60,7 @@ public class GlobalErrorAttributesHandler extends DefaultErrorAttributes { // 如果返回的异常是CommonException,则按CommonException响应的内容进行返回 Throwable throwable = this.getError(webRequest); if (ObjectUtil.isNotEmpty(throwable)) { - if (throwable instanceof CommonException) { - CommonException commonException = (CommonException) throwable; + if (throwable instanceof CommonException commonException) { return BeanUtil.beanToMap(CommonResult.error(commonException.getMsg())); } else { return BeanUtil.beanToMap(CommonResult.get(HttpStatus.HTTP_INTERNAL_ERROR, "服务器异常,请求地址:" + diff --git a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewController.java b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewController.java index 1eb623b9..540cecdd 100644 --- a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewController.java +++ b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewController.java @@ -14,12 +14,15 @@ package vip.xiaonuo.core.handler; import cn.hutool.core.util.ObjectUtil; import jakarta.servlet.http.HttpServletRequest; +import jakarta.servlet.http.HttpServletResponse; import lombok.extern.slf4j.Slf4j; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.pojo.CommonResult; +import java.io.IOException; + /** * 全局异常页面处理器,覆盖默认的Whitelabel Error Page * @@ -37,7 +40,7 @@ public class GlobalErrorViewController { * @date 2022/2/11 16:11 **/ @RequestMapping("/errorView") - public CommonResult globalError(HttpServletRequest request) { + public CommonResult globalError(HttpServletRequest request, HttpServletResponse response) throws IOException { CommonResult commonResult = new CommonResult<>(404, "路径不存在", null); Object model = request.getAttribute("model"); if(ObjectUtil.isNotEmpty(model)) { diff --git a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewHandler.java b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewHandler.java index 29fe8cd5..c1eeb10d 100644 --- a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewHandler.java +++ b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalErrorViewHandler.java @@ -16,7 +16,6 @@ import cn.hutool.core.convert.Convert; import cn.hutool.core.util.ObjectUtil; import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; -import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.autoconfigure.web.ServerProperties; import org.springframework.boot.autoconfigure.web.servlet.error.BasicErrorController; import org.springframework.http.HttpStatus; @@ -38,7 +37,7 @@ import java.util.Map; @RestController public class GlobalErrorViewHandler extends BasicErrorController { - public GlobalErrorViewHandler(@Autowired(required = false) ServerProperties serverProperties) { + public GlobalErrorViewHandler(ServerProperties serverProperties) { super(new GlobalErrorAttributesHandler(), serverProperties.getError()); } diff --git a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalExceptionUtil.java b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalExceptionUtil.java index 8b00d16d..fba33e8f 100644 --- a/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalExceptionUtil.java +++ b/snowy-web-app/src/main/java/vip/xiaonuo/core/handler/GlobalExceptionUtil.java @@ -16,8 +16,6 @@ import cn.dev33.satoken.exception.SaTokenException; import cn.hutool.core.text.StrPool; import cn.hutool.core.util.ObjectUtil; import cn.hutool.http.HttpStatus; -import jakarta.validation.ConstraintViolation; -import jakarta.validation.ConstraintViolationException; import lombok.extern.slf4j.Slf4j; import org.apache.commons.collections4.CollectionUtils; import org.apache.commons.lang3.StringUtils; @@ -39,6 +37,8 @@ import vip.xiaonuo.common.exception.CommonException; import vip.xiaonuo.common.pojo.CommonResult; import vip.xiaonuo.common.util.CommonServletUtil; +import javax.validation.ConstraintViolation; +import javax.validation.ConstraintViolationException; import java.util.Set; import java.util.stream.Collectors; @@ -127,7 +127,7 @@ public class GlobalExceptionUtil { log.error(">>> 数据操作异常:", e); commonResult = CommonResult.error("数据操作异常"); } - } else { + }else { log.error(">>> 数据操作异常:", e); commonResult = CommonResult.error("数据操作异常"); } diff --git a/snowy-web-app/src/main/resources/_sql/snowy_mysql.sql b/snowy-web-app/src/main/resources/_sql/snowy_mysql.sql index 40a12dab..f9533061 100644 --- a/snowy-web-app/src/main/resources/_sql/snowy_mysql.sql +++ b/snowy-web-app/src/main/resources/_sql/snowy_mysql.sql @@ -21,12 +21,40 @@ CREATE TABLE `AUTH_THIRD_USER` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '三方用户' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '三方用户' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of AUTH_THIRD_USER -- ---------------------------- +-- ---------------------------- +-- Table structure for BIZ_NOTICE +-- ---------------------------- +DROP TABLE IF EXISTS `BIZ_NOTICE`; +CREATE TABLE `BIZ_NOTICE` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '主键', + `TITLE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '标题', + `IMAGE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '封面图', + `CONTENT` text CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '内容', + `DIGEST` varchar(300) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '摘要', + `TYPE` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '类型', + `PLACE` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '发布位置', + `STATUS` varchar(10) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '状态', + `SORT_CODE` int(11) NULL DEFAULT NULL COMMENT '排序', + `REMARK` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '备注', + `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '更新时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '更新用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '通知公告' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of BIZ_NOTICE +-- ---------------------------- + -- ---------------------------- -- Table structure for CLIENT_RELATION -- ---------------------------- @@ -38,7 +66,7 @@ CREATE TABLE `CLIENT_RELATION` ( `CATEGORY` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '分类', `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '关系' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '关系' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of CLIENT_RELATION @@ -95,12 +123,53 @@ CREATE TABLE `CLIENT_USER` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = 'C端用户' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = 'C端用户' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of CLIENT_USER -- ---------------------------- +-- ---------------------------- +-- Table structure for CLIENT_USER_EXT +-- ---------------------------- +DROP TABLE IF EXISTS `CLIENT_USER_EXT`; +CREATE TABLE `CLIENT_USER_EXT` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL, + `USER_ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '用户id', + `SOURCE_FROM_TYPE` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '来源类别', + `PASSWORD_UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '密码修改日期', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = 'C端用户扩展' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of CLIENT_USER_EXT +-- ---------------------------- + +-- ---------------------------- +-- Table structure for CLIENT_USER_PASSWORD +-- ---------------------------- +DROP TABLE IF EXISTS `CLIENT_USER_PASSWORD`; +CREATE TABLE `CLIENT_USER_PASSWORD` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL, + `USER_ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '用户id', + `PASSWORD` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '密码', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = 'C端用户密码' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of CLIENT_USER_PASSWORD +-- ---------------------------- + -- ---------------------------- -- Table structure for DEV_CONFIG -- ---------------------------- @@ -119,19 +188,19 @@ CREATE TABLE `DEV_CONFIG` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '配置' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '配置' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_CONFIG -- ---------------------------- INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967455', 'SNOWY_SYS_LOGO', '', 'SYS_BASE', '系统LOGO', 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967456', 'SNOWY_SYS_NAME', 'Snowy', 'SYS_BASE', '系统名称', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967457', 'SNOWY_SYS_VERSION', 'V2.0.0', 'SYS_BASE', '系统版本', 3, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967457', 'SNOWY_SYS_VERSION', 'V3.0.0', 'SYS_BASE', '系统版本', 3, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967458', 'SNOWY_SYS_COPYRIGHT', 'Snowy ©2022 Created by 小诺开源技术', 'SYS_BASE', '系统版权', 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967459', 'SNOWY_SYS_COPYRIGHT_URL', 'https://www.xiaonuo.vip', 'SYS_BASE', '系统版权链接地址', 5, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967462', 'SNOWY_SYS_DEFAULT_CAPTCHA_OPEN', 'false', 'SYS_BASE', '登录验证码开关', 8, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967462', 'SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_B', 'B端全局验证码开关', 'SYS_BASE', 'B端全局验证码开关', 8, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967463', 'SNOWY_SYS_DEFAULT_FILE_ENGINE', 'LOCAL', 'SYS_BASE', '默认文件存储引擎', 9, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967464', 'SNOWY_SYS_DEFAULT_PASSWORD', '123456', 'SYS_BASE', '默认用户密码', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967464', 'SNOWY_SYS_DEFAULT_PASSWORD_FOR_B', '123456', 'PASSWORD_STRATEGY_FOR_B', 'B端默认用户密码', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967465', 'SNOWY_SYS_DEFAULT_DESCRRIPTION', 'Snowy是一款国内首例国产密码算法加密框架,采用Vue3.0+AntDesignVue3.0+SpringBoot2.8前后分离技术打造,技术框架与密码的结合,让前后分离‘密’不可分!', 'SYS_BASE', '系统描述', 11, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967466', 'SNOWY_SYS_DEFAULT_WORKBENCH_DATA', '{\"shortcut\":[{\"id\":\"1548901111999770526\",\"title\":\"系统首页\",\"icon\":\"home-outlined\",\"path\":\"/index\"}]}', 'SYS_BASE', '系统默认工作台数据', 12, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967558', 'SNOWY_THIRD_GITEE_CLIENT_ID', 'GiteeClientId', 'THIRD_GITEE', 'GiteeClientId', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); @@ -158,18 +227,16 @@ INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967578', 'SNOWY_EMAIL_LOCAL_FROM' INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967579', 'SNOWY_EMAIL_LOCAL_PASSWORD', '本地邮件发件密码', 'EMAIL_LOCAL', '本地邮件发件密码', 34, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967580', 'SNOWY_EMAIL_TENCENT_SECRET_ID', '腾讯云邮件SecretId', 'EMAIL_TENCENT', '腾讯云邮件SecretId', 35, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967581', 'SNOWY_EMAIL_TENCENT_SECRET_KEY', '腾讯云邮件SecretKey', 'EMAIL_TENCENT', '腾讯云邮件SecretKey', 36, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967582', 'SNOWY_EMAIL_TENCENT_REGION_ID', '腾讯云邮件RegionId', 'EMAIL_TENCENT', '腾讯云邮件RegionId', 37, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967582', 'SNOWY_EMAIL_TENCENT_FROM', '默认发送账号', 'EMAIL_TENCENT', '默认发送账号', 37, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967583', 'SNOWY_EMAIL_ALIYUN_ACCESS_KEY_ID', '阿里云邮件AccessKeyId', 'EMAIL_ALIYUN', '阿里云邮件AccessKeyId', 38, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967584', 'SNOWY_EMAIL_ALIYUN_ACCESS_KEY_SECRET', '阿里云邮件AccessKeySecret', 'EMAIL_ALIYUN', '阿里云邮件AccessKeySecret', 39, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967585', 'SNOWY_EMAIL_ALIYUN_REGION_ID', '阿里云邮件RegionId', 'EMAIL_ALIYUN', '阿里云邮件RegionId', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967585', 'SNOWY_EMAIL_ALIYUN_FROM', '默认发送账号', 'EMAIL_ALIYUN', '默认发送账号', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967586', 'SNOWY_SMS_TENCENT_SECRET_ID', '腾讯云短信SecretId', 'SMS_TENCENT', '腾讯云短信SecretId', 41, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967587', 'SNOWY_SMS_TENCENT_SECRET_KEY', '腾讯云短信SecretKey', 'SMS_TENCENT', '腾讯云短信SecretKey', 42, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967588', 'SNOWY_SMS_TENCENT_REGION_ID', '腾讯云短信RegionId', 'SMS_TENCENT', '腾讯云短信RegionId', 43, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967589', 'SNOWY_SMS_TENCENT_DEFAULT_SDK_APP_ID', '腾讯云短信默认SdkAppId', 'SMS_TENCENT', '腾讯云短信默认SdkAppId', 44, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967590', 'SNOWY_SMS_TENCENT_DEFAULT_SIGN_NAME', '腾讯云短信默认签名', 'SMS_TENCENT', '腾讯云短信默认签名', 45, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967591', 'SNOWY_SMS_ALIYUN_ACCESS_KEY_ID', '阿里云短信AccessKeyId', 'SMS_ALIYUN', '阿里云短信AccessKeyId', 46, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967592', 'SNOWY_SMS_ALIYUN_ACCESS_KEY_SECRET', '阿里云短信AccessKeySecret', 'SMS_ALIYUN', '阿里云短信AccessKeySecret', 47, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967593', 'SNOWY_SMS_ALIYUN_END_POINT', '阿里云短信EndPoint', 'SMS_ALIYUN', '阿里云短信EndPoint', 48, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967594', 'SNOWY_SMS_ALIYUN_DEFAULT_SIGN_NAME', '阿里云短信默认签名', 'SMS_ALIYUN', '阿里云短信默认签名', 49, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967595', 'SNOWY_SMS_XIAONUO_ACCESS_KEY_ID', '小诺短信账号Id', 'SMS_XIAONUO', '小诺短信账号Id', 50, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362967596', 'SNOWY_SMS_XIAONUO_ACCESS_KEY_SECRET', '小诺短信账号Secret', 'SMS_XIAONUO', '小诺短信账号', 51, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); @@ -180,6 +247,96 @@ INSERT INTO `DEV_CONFIG` VALUES ('1554740179362968001', 'SNOWY_EMAIL_LOCAL_SMTP_ INSERT INTO `DEV_CONFIG` VALUES ('1554740179362968002', 'SNOWY_EMAIL_LOCAL_AUTH', 'true', 'EMAIL_LOCAL', '本地邮件是否需要用户名密码验证', 82, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362968003', 'SNOWY_EMAIL_LOCAL_SSL_ENABLE', 'true', 'EMAIL_LOCAL', '本地邮件是否使用SSL安全连接', 83, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `DEV_CONFIG` VALUES ('1554740179362968004', 'SNOWY_EMAIL_LOCAL_STARTTLS_ENABLE', 'true', 'EMAIL_LOCAL', '本地邮件是否使用STARTTLS安全连接', 84, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755202', 'SNOWY_SYS_DEFAULT_PUSH_ENGINE', 'DINGTALK', 'SYS_BASE', '默认消息推送引擎', 85, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755203', 'SNOWY_SYS_DEFAULT_SMS_ENGINE', 'XIAONUO', 'SYS_BASE', '默认短信发送引擎', 86, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755204', 'SNOWY_SYS_DEFAULT_EMAIL_ENGINE', 'LOCAL', 'SYS_BASE', '默认邮件发送引擎', 87, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755205', 'SNOWY_SYS_DEFAULT_CAPTCHA_OPEN_FLAG_FOR_C', 'true', 'SYS_BASE', 'C端全局验证码开关', 88, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755206', 'SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_B', '5', 'SYS_BASE', 'B端验证码失效时间', 89, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755207', 'SNOWY_SYS_DEFAULT_CAPTCHA_EXPIRED_DURATION_FOR_C', '5', 'SYS_BASE', 'C端验证码失效时间', 90, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755208', 'SNOWY_PUSH_DINGTALK_SIGN', '钉钉消息推送签名', 'PUSH_DINGTALK', '钉钉消息推送签名', 91, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755209', 'SNOWY_PUSH_DINGTALK_TOKEN_ID', '钉钉消息推送TOKENID', 'PUSH_DINGTALK', '钉钉消息推送TOKENID', 92, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755210', 'SNOWY_PUSH_FEISHU_TOKEN_ID', '飞书消息推送TOKENID', 'PUSH_FEISHU', '飞书消息推送TOKENID', 93, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755211', 'SNOWY_PUSH_WORKWECHAT_TOKEN_ID', '企业微信消息推送TOKENID', 'PUSH_WORKWECHAT', '企业微信消息推送TOKENID', 94, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755212', 'SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B', '{\"subject\":\"注册账号成功提醒\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您的账号注册成功,感谢您加入{sysName}。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端注册账号成功邮件消息模板内容', 95, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755213', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_B', '{\"subject\":\"登录系统验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在登录系统, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端登录系统验证码邮件消息模板内容', 96, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755214', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B', '{\"subject\":\"修改密码验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在修改密码, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端修改密码验证码邮件消息模板内容', 97, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755215', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B', '{\"subject\":\"重置密码验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在重置密码, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端重置密码验证码邮件消息模板内容', 98, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755216', 'SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B', '{\"subject\":\"重置密码成功提醒\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您的密码已经重置,重置后的密码为:{userNewPassword},如非本人操作请立即修账户密码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端重置密码成功邮件消息模板内容', 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755217', 'SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B', '{\"subject\":\"密码即将到期提醒\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您的密码即将过期,请尽快修改密码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端密码即将到期邮件消息模板内容', 100, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755218', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_B', '{\"subject\":\"绑定邮箱验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在绑定邮箱, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端绑定邮箱验证码邮件消息模板内容', 101, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755219', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_B', '{\"subject\":\"修改绑定邮箱验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在修改绑定邮箱, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_B', 'B端修改绑定邮箱验证码邮件消息模板内容', 102, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755220', 'SNOWY_EMAIL_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C', '{\"subject\":\"注册账号成功提醒\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您的账号注册成功,感谢您加入{sysName}。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端注册账号成功邮件消息模板内容', 103, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755221', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_LOGIN_FOR_C', '{\"subject\":\"登录系统验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在登录系统, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端登录系统验证码邮件消息模板内容', 104, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755222', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C', '{\"subject\":\"修改密码验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在修改密码, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端修改密码验证码邮件消息模板内容', 105, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755223', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_C', '{\"subject\":\"重置密码验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在重置密码, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端重置密码验证码邮件消息模板内容', 106, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755224', 'SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C', '{\"subject\":\"重置密码成功提醒\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您的密码已经重置,重置后的密码为:{userNewPassword},如非本人操作请立即修账户密码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端重置密码成功邮件消息模板内容', 107, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755225', 'SNOWY_EMAIL_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C', '{\"subject\":\"密码即将到期提醒\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您的密码即将过期,请尽快修改密码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端密码即将到期邮件消息模板内容', 108, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755226', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_BINDING_EMAIL_FOR_C', '{\"subject\":\"绑定邮箱验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在绑定邮箱, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端绑定邮箱验证码邮件消息模板内容', 109, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755227', 'SNOWY_EMAIL_TEMPLATE_VALID_CODE_UPDATE_BINDING_EMAIL_FOR_C', '{\"subject\":\"修改绑定邮箱验证码\",\"content\":\"
\\n \\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n {sysName}\\n
\\n
\\n

\\n \\n 亲爱的用户:{userEmail}\\n \\n

\\n

您正在修改绑定邮箱, 您的验证码为:{validCode},该验证码{validTime}分钟内有效,请尽快验证并保管好验证码。

\\n

{sysName}

\\n

{sysNowTime}

\\n
\\n
\\n
\"}', 'EMAIL_TEMPLATE_FOR_C', 'C端修改绑定邮箱验证码邮件消息模板内容', 110, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755228', 'SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_B', '{\"code\":\"\",\"content\":\"您的账号注册成功,感谢您加入${sysName}。\"}', 'SMS_TEMPLATE_FOR_B', 'B端注册账号成功短信消息模板编码', 111, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755229', 'SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_B', '{\"code\":\"\",\"content\":\"您正在登录系统, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_B', 'B端登录系统验证码短信消息模板编码', 112, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755230', 'SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_B', '{\"code\":\"\",\"content\":\"您正在修改密码, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_B', 'B端修改密码验证码短信消息模板编码', 113, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755231', 'SNOWY_SMS_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_B', '{\"code\":\"\",\"content\":\"您正在重置密码, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_B', 'B端重置密码验证码短信消息模板编码', 114, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755232', 'SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_B', '{\"code\":\"\",\"content\":\"您的密码已经重置,重置后的密码为:${userNewPassword},如非本人操作请立即修账户密码。\"}', 'SMS_TEMPLATE_FOR_B', 'B端重置密码成功短信消息模板编码', 115, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755233', 'SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_B', '{\"code\":\"\",\"content\":\"您的密码即将过期,请尽快修改密码。\"}', 'SMS_TEMPLATE_FOR_B', 'B端密码即将到期短信消息模板编码', 116, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755234', 'SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_B', '{\"code\":\"\",\"content\":\"您正在绑定手机, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_B', 'B端绑定手机验证码短信消息模板编码', 117, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755235', 'SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_B', '{\"code\":\"\",\"content\":\"您正在修改绑定手机, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_B', 'B端修改绑定手机验证码短信消息模板编码', 118, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755236', 'SNOWY_SMS_TEMPLATE_NOTICE_REGISTER_SUCCESS_FOR_C', '{\"code\":\"\",\"content\":\"您的账号注册成功,感谢您加入${sysName}。\"}', 'SMS_TEMPLATE_FOR_C', 'C端注册账号成功短信消息模板编码', 119, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755237', 'SNOWY_SMS_TEMPLATE_VALID_CODE_LOGIN_FOR_C', '{\"code\":\"\",\"content\":\"您正在登录系统, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_C', 'C端登录系统验证码短信消息模板编码', 120, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755238', 'SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_PASSWORD_FOR_C', '{\"code\":\"\",\"content\":\"您正在修改密码, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_C', 'C端修改密码验证码短信消息模板编码', 121, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755239', 'SNOWY_SMS_TEMPLATE_VALID_CODE_RESET_PASSWORD_FOR_C', '{\"code\":\"\",\"content\":\"您正在重置密码, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_C', 'C端重置密码验证码短信消息模板编码', 122, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755240', 'SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_RESET_SUCCESS_FOR_C', '{\"code\":\"\",\"content\":\"您的密码已经重置,重置后的密码为:${userNewPassword},如非本人操作请立即修账户密码。\"}', 'SMS_TEMPLATE_FOR_C', 'C端重置密码成功短信消息模板编码', 123, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755241', 'SNOWY_SMS_TEMPLATE_NOTICE_PASSWORD_EXPIRED_FOR_C', '{\"code\":\"\",\"content\":\"您的密码即将过期,请尽快修改密码。\"}', 'SMS_TEMPLATE_FOR_C', 'C端密码即将到期短信消息模板编码', 124, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755242', 'SNOWY_SMS_TEMPLATE_VALID_CODE_BINDING_PHONE_FOR_C', '{\"code\":\"\",\"content\":\"您正在绑定手机, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_C', 'C端绑定手机验证码短信消息模板编码', 125, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755243', 'SNOWY_SMS_TEMPLATE_VALID_CODE_UPDATE_BINDING_PHONE_FOR_C', '{\"code\":\"\",\"content\":\"您正在修改绑定手机, 您的验证码为:${validCode},该验证码${validTime}分钟内有效,请尽快验证并保管好验证码。\"}', 'SMS_TEMPLATE_FOR_C', 'C端修改绑定手机验证码短信消息模板编码', 126, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755244', 'SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_B', '5', 'LOGIN_STRATEGY_FOR_B', 'B端连续登录失败持续时间', 127, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755245', 'SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_B', '5', 'LOGIN_STRATEGY_FOR_B', 'B端连续登录失败次数', 128, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755246', 'SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_B', '5', 'LOGIN_STRATEGY_FOR_B', 'B端连续登录失败锁定时间', 129, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755247', 'SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_B', 'true', 'LOGIN_STRATEGY_FOR_B', 'B端是否允许手机号登录', 130, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755248', 'SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_B', 'NOT_ALLOW_LOGIN', 'LOGIN_STRATEGY_FOR_B', 'B端手机号无对应用户时策略', 131, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755249', 'SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_B', 'true', 'LOGIN_STRATEGY_FOR_B', 'B端是否允许邮箱登录', 132, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755250', 'SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_B', 'NOT_ALLOW_LOGIN', 'LOGIN_STRATEGY_FOR_B', 'B端邮箱无对应用户时策略', 133, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755251', 'SNOWY_SYS_DEFAULT_NEW_USER_ORG_FOR_B', '1543842934270394368', 'REGISTER_STRATEGY_FOR_B', 'B端新用户默认机构', 144, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755252', 'SNOWY_SYS_DEFAULT_NEW_USER_POSITION_FOR_B', '1543899639134019583', 'REGISTER_STRATEGY_FOR_B', 'B端新用户默认职位', 145, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755253', 'SNOWY_SYS_DEFAULT_NEW_USER_ROLE_FOR_B', '1570687866138206208', 'REGISTER_STRATEGY_FOR_B', 'B端新用户默认角色', 146, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755254', 'SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_DURATION_FOR_C', '5', 'LOGIN_STRATEGY_FOR_C', 'C端连续登录失败持续时间', 134, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755255', 'SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_TIMES_FOR_C', '5', 'LOGIN_STRATEGY_FOR_C', 'C端连续登录失败次数', 135, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755256', 'SNOWY_SYS_DEFAULT_CONTINUOUS_LOGIN_FAIL_LOCK_DURATION_FOR_C', '5', 'LOGIN_STRATEGY_FOR_C', 'C端连续登录失败锁定时间', 136, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755257', 'SNOWY_SYS_DEFAULT_ALLOW_PHONE_LOGIN_FLAG_FOR_C', 'true', 'LOGIN_STRATEGY_FOR_C', 'C端是否允许手机号登录', 137, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755258', 'SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_PHONE_FOR_C', 'NOT_ALLOW_LOGIN', 'LOGIN_STRATEGY_FOR_C', 'C端手机号无对应用户时策略', 138, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755259', 'SNOWY_SYS_DEFAULT_ALLOW_EMAIL_LOGIN_FLAG_FOR_C', 'true', 'LOGIN_STRATEGY_FOR_C', 'C端是否允许邮箱登录', 139, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755260', 'SNOWY_SYS_DEFAULT_STRATEGY_WHEN_NO_USER_WITH_EMAIL_FOR_C', 'NOT_ALLOW_LOGIN', 'LOGIN_STRATEGY_FOR_C', 'C端邮箱无对应用户时策略', 140, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755261', 'SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_B', 'true', 'REGISTER_STRATEGY_FOR_B', 'B端是否允许注册', 141, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755262', 'SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_B', 'false', 'REGISTER_STRATEGY_FOR_B', 'B端注册后是否需要绑定手机号', 142, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755263', 'SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_B', 'false', 'REGISTER_STRATEGY_FOR_B', 'B端注册后是否需要绑定邮箱', 143, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755264', 'SNOWY_SYS_DEFAULT_ALLOW_REGISTER_FLAG_FOR_C', 'true', 'REGISTER_STRATEGY_FOR_C', 'C端是否允许注册', 147, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755265', 'SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_PHONE_FOR_C', 'false', 'REGISTER_STRATEGY_FOR_C', 'C端注册后是否需要绑定手机号', 148, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755266', 'SNOWY_SYS_DEFAULT_REGISTER_NEED_BIND_EMAIL_FOR_C', 'false', 'REGISTER_STRATEGY_FOR_C', 'C端注册后是否需要绑定邮箱', 149, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755267', 'SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_B', 'OLD', 'PASSWORD_STRATEGY_FOR_B', 'B端密码修改验证方式', 150, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755268', 'SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_B', '6', 'PASSWORD_STRATEGY_FOR_B', 'B端密码最小长度', 151, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755269', 'SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_B', '20', 'PASSWORD_STRATEGY_FOR_B', 'B端密码最大长度', 152, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755270', 'SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_B', 'REG0', 'PASSWORD_STRATEGY_FOR_B', 'B端密码复杂度', 153, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755271', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_B', '3', 'PASSWORD_STRATEGY_FOR_B', 'B端密码不能连续存在相同字符个数', 154, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755272', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_B', 'true', 'PASSWORD_STRATEGY_FOR_B', 'B端密码不能包含用户信息开关', 155, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755273', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_B', 'true', 'PASSWORD_STRATEGY_FOR_B', 'B端密码不能使用历史密码开关', 156, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755274', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_B', '3', 'PASSWORD_STRATEGY_FOR_B', 'B端密码不能使用历史密码范围个数', 157, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755275', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_B', 'true', 'PASSWORD_STRATEGY_FOR_B', 'B端密码不能使用弱密码库中密码开关', 158, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755276', 'SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_B', 'xiaonuo,xiaonuoark', 'PASSWORD_STRATEGY_FOR_B', 'B端密码自定义额外弱密码库', 159, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755277', 'SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_B', '30', 'PASSWORD_STRATEGY_FOR_B', 'B端密码有效期天数', 160, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755278', 'SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_B', '3', 'PASSWORD_STRATEGY_FOR_B', 'B端密码过期前提醒天数', 161, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755279', 'SNOWY_SYS_DEFAULT_PASSWORD_FOR_C', '123456', 'PASSWORD_STRATEGY_FOR_C', 'C端密码默认密码', 162, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755280', 'SNOWY_SYS_DEFAULT_PASSWORD_UPDATE_VALID_TYPE_FOR_C', 'OLD', 'PASSWORD_STRATEGY_FOR_C', 'C端密码修改验证方式', 163, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755281', 'SNOWY_SYS_DEFAULT_PASSWORD_MIN_LENGTH_FOR_C', '6', 'PASSWORD_STRATEGY_FOR_C', 'C端密码最小长度', 164, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755282', 'SNOWY_SYS_DEFAULT_PASSWORD_MAX_LENGTH_FOR_C', '20', 'PASSWORD_STRATEGY_FOR_C', 'C端密码最大长度', 165, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755283', 'SNOWY_SYS_DEFAULT_PASSWORD_COMPLEXITY_FOR_C', 'REG0', 'PASSWORD_STRATEGY_FOR_C', 'C端密码复杂度', 166, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755284', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTINUOUS_SAME_CHARACTER_LENGTH_FOR_C', '3', 'PASSWORD_STRATEGY_FOR_C', 'C端密码不能连续存在相同字符个数', 167, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755285', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_CONTAINS_USER_INFO_FLAG_FOR_C', 'true', 'PASSWORD_STRATEGY_FOR_C', 'C端密码不能包含用户信息开关', 168, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755286', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_FLAG_FOR_C', 'true', 'PASSWORD_STRATEGY_FOR_C', 'C端密码不能使用历史密码开关', 169, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755287', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_HISTORY_COUNT_FOR_C', '3', 'PASSWORD_STRATEGY_FOR_C', 'C端密码不能使用历史密码范围个数', 170, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755288', 'SNOWY_SYS_DEFAULT_PASSWORD_NOT_ALLOW_USE_WEAK_FLAG_FOR_C', 'true', 'PASSWORD_STRATEGY_FOR_C', 'C端密码不能使用弱密码库中密码开关', 171, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755289', 'SNOWY_SYS_DEFAULT_PASSWORD_DEFINE_WEAK_DATABASE_FOR_C', 'xiaonuo,xiaonuoark', 'PASSWORD_STRATEGY_FOR_C', 'C端密码自定义额外弱密码库', 172, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755290', 'SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_DAYS_FOR_C', '30', 'PASSWORD_STRATEGY_FOR_C', 'C端密码有效期天数', 173, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_CONFIG` VALUES ('1908870094824755291', 'SNOWY_SYS_DEFAULT_PASSWORD_EXPIRED_NOTICE_DAYS_FOR_C', '3', 'PASSWORD_STRATEGY_FOR_C', 'C端密码过期前提醒天数', 174, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -- ---------------------------- -- Table structure for DEV_DICT @@ -190,6 +347,7 @@ CREATE TABLE `DEV_DICT` ( `PARENT_ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '父id', `DICT_LABEL` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '字典文字', `DICT_VALUE` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '字典值', + `CODE` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '编码', `CATEGORY` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '分类', `SORT_CODE` int(11) NULL DEFAULT NULL COMMENT '排序码', `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', @@ -199,172 +357,190 @@ CREATE TABLE `DEV_DICT` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '字典' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '字典' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_DICT -- ---------------------------- -INSERT INTO `DEV_DICT` VALUES ('1543839774776291330', '0', '用户性别类型', 'GENDER', 'FRM', 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1543839901037424642', '1543839774776291330', '男', '男', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1543840033980084226', '1543839774776291330', '女', '女', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1543860103661809666', '0', '系统菜单类型', 'MENU_TYPE', 'FRM', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1543860239020388354', '1543860103661809666', '目录', 'CATALOG', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1543860305508495361', '1543860103661809666', '菜单', 'MENU', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1543860423485878274', '1543860103661809666', '内链', 'IFRAME', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1543860477512708098', '1543860103661809666', '外链', 'LINK', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1544329873407946753', '0', '系统通用状态', 'COMMON_STATUS', 'FRM', 3, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1545397556652027906', '1544329873407946753', '启用', 'ENABLE', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1545397556652027907', '1544329873407946753', '停用', 'DISABLED', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547207669861064706', '0', '系统角色分类', 'ROLE_CATEGORY', 'FRM', 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547207891009937409', '1547207669861064706', '全局', 'GLOBAL', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547207990075203585', '1547207669861064706', '机构', 'ORG', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547227094995705858', '0', '系统机构分类', 'ORG_CATEGORY', 'FRM', 5, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547227670693289985', '1547227094995705858', '部门', 'DEPT', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547228161267474434', '1547227094995705858', '公司', 'COMPANY', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547409689434742786', '0', '系统职位分类', 'POSITION_CATEGORY', 'FRM', 6, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547409794837602305', '1547409689434742786', '高层', 'HIGH', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547409844779180033', '1547409689434742786', '中层', 'MIDDLE', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547409906406088706', '1547409689434742786', '基层', 'LOW', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658113', '0', '用户民族类型', 'NATION', 'FRM', 7, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658114', '1547641470701658113', '汉族', '汉族', 'FRM', 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658115', '1547641470701658113', '壮族', '壮族', 'FRM', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658116', '1547641470701658113', '回族', '回族', 'FRM', 3, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658117', '1547641470701658113', '满族', '满族', 'FRM', 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658118', '1547641470701658113', '维吾尔族', '维吾尔族', 'FRM', 5, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658119', '1547641470701658113', '苗族', '苗族', 'FRM', 6, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658120', '1547641470701658113', '彝族', '彝族', 'FRM', 7, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658121', '1547641470701658113', '土家族', '土家族', 'FRM', 8, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658122', '1547641470701658113', '藏族', '藏族', 'FRM', 9, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658123', '1547641470701658113', '蒙古族', '蒙古族', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658124', '1547641470701658113', '侗族', '侗族', 'FRM', 11, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658125', '1547641470701658113', '布依族', '布依族', 'FRM', 12, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658126', '1547641470701658113', '瑶族', '瑶族', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658127', '1547641470701658113', '白族', '白族', 'FRM', 14, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658128', '1547641470701658113', '朝鲜族', '朝鲜族', 'FRM', 15, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658129', '1547641470701658113', '哈尼族', '哈尼族', 'FRM', 16, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658130', '1547641470701658113', '黎族', '黎族', 'FRM', 17, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658131', '1547641470701658113', '哈萨克族', '哈萨克族', 'FRM', 18, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658132', '1547641470701658113', '傣族', '傣族', 'FRM', 19, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658133', '1547641470701658113', '畲族', '畲族', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658134', '1547641470701658113', '傈僳族', '傈僳族', 'FRM', 21, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658135', '1547641470701658113', '东乡族', '东乡族', 'FRM', 22, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658136', '1547641470701658113', '仡佬族', '仡佬族', 'FRM', 23, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658137', '1547641470701658113', '拉祜族', '拉祜族', 'FRM', 24, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658138', '1547641470701658113', '佤族', '佤族', 'FRM', 25, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658139', '1547641470701658113', '水族', '水族', 'FRM', 26, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658140', '1547641470701658113', '纳西族', '纳西族', 'FRM', 27, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658141', '1547641470701658113', '羌族', '羌族', 'FRM', 28, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658142', '1547641470701658113', '土族', '土族', 'FRM', 29, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658143', '1547641470701658113', '仫佬族', '仫佬族', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658144', '1547641470701658113', '锡伯族', '锡伯族', 'FRM', 31, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658145', '1547641470701658113', '柯尔克孜族', '柯尔克孜族', 'FRM', 32, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658146', '1547641470701658113', '景颇族', '景颇族', 'FRM', 33, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658147', '1547641470701658113', '达斡尔族', '达斡尔族', 'FRM', 34, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658148', '1547641470701658113', '撒拉族', '撒拉族', 'FRM', 35, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658149', '1547641470701658113', '布朗族', '布朗族', 'FRM', 36, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658150', '1547641470701658113', '毛南族', '毛南族', 'FRM', 37, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658151', '1547641470701658113', '塔吉克族', '塔吉克族', 'FRM', 38, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658152', '1547641470701658113', '普米族', '普米族', 'FRM', 39, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658153', '1547641470701658113', '阿昌族', '阿昌族', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658154', '1547641470701658113', '怒族', '怒族', 'FRM', 41, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658155', '1547641470701658113', '鄂温克族', '鄂温克族', 'FRM', 42, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658156', '1547641470701658113', '京族', '京族', 'FRM', 43, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658157', '1547641470701658113', '基诺族', '基诺族', 'FRM', 44, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658158', '1547641470701658113', '德昂族', '德昂族', 'FRM', 45, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658159', '1547641470701658113', '保安族', '保安族', 'FRM', 46, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658160', '1547641470701658113', '俄罗斯族', '俄罗斯族', 'FRM', 47, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658161', '1547641470701658113', '裕固族', '裕固族', 'FRM', 48, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658162', '1547641470701658113', '乌孜别克族', '乌孜别克族', 'FRM', 49, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658163', '1547641470701658113', '门巴族', '门巴族', 'FRM', 50, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658164', '1547641470701658113', '鄂伦春族', '鄂伦春族', 'FRM', 51, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658165', '1547641470701658113', '独龙族', '独龙族', 'FRM', 52, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658166', '1547641470701658113', '赫哲族', '赫哲族', 'FRM', 53, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658167', '1547641470701658113', '高山族', '高山族', 'FRM', 54, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658168', '1547641470701658113', '珞巴族', '珞巴族', 'FRM', 55, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1547641470701658169', '1547641470701658113', '塔塔尔族 ', '塔塔尔族 ', 'FRM', 56, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1549019269252104194', '0', '登录设备类型', 'AUTH_DEVICE_TYPE', 'FRM', 8, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1549019748883349506', '1549019269252104194', 'PC端', 'PC', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1549019813924421634', '1549019269252104194', '移动端', 'APP', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1549019864537088002', '1549019269252104194', '小程序', 'MINI', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554678566166323202', '0', '系统字典分类', 'DICT_CATEGORY', 'FRM', 9, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554678761742524417', '1554678566166323202', '框架', 'FRM', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554678863617974273', '1554678566166323202', '业务', 'BIZ', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554679788378120194', '0', '短信发送引擎', 'SMS_ENGINE', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554679872054484993', '1554679788378120194', '阿里云', 'ALIYUN', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554679958398427138', '1554679788378120194', '腾讯云', 'TENCENT', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554679958398427139', '1554679788378120194', '小诺方舟', 'XIAONUO', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554726376265744386', '0', '文件上传引擎', 'FILE_ENGINE', 'FRM', 11, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554726639571566593', '1554726376265744386', '本地', 'LOCAL', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554726696068841474', '1554726376265744386', '阿里云', 'ALIYUN', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554726762338844674', '1554726376265744386', '腾讯云', 'TENCENT', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1554726830844411905', '1554726376265744386', 'MINIO', 'MINIO', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1556317797993218049', '0', '邮件发送引擎', 'EMAIL_ENGINE', 'FRM', 12, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1556319651447767041', '1556317797993218049', '本地', 'LOCAL', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1556319726962016258', '1556317797993218049', '阿里云', 'ALIYUN', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1556319786349166593', '1556317797993218049', '腾讯云', 'TENCENT', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1559942543251079169', '0', '系统通用开关', 'COMMON_SWITCH', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1559942706694717442', '1559942543251079169', '开', 'true', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1559942777674924034', '1559942543251079169', '关', 'false', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560306347516461058', '0', '用户证件类型', 'IDCARD_TYPE', 'FRM', 14, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560306502135283714', '1560306347516461058', '身份证', '身份证', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560306768913989633', '1560306347516461058', '出生证', '出生证', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560307009365049346', '1560306347516461058', '军官证', '军官证', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560307180937248769', '1560306347516461058', '护照', '护照', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309358598914050', '0', '通用文化程度', 'CULTURE_LEVEL', 'FRM', 15, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309494892822530', '1560309358598914050', '小学', '小学', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309602136981505', '1560309358598914050', '中学', '中学', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309700136894465', '1560309358598914050', '高中', '高中', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309744118366209', '1560309358598914050', '中专', '中专', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309783037313026', '1560309358598914050', '大专', '大专', 'FRM', 50, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309816423972866', '1560309358598914050', '本科', '本科', 'FRM', 60, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560309855661686785', '1560309358598914050', '硕士研究生', '硕士研究生', 'FRM', 70, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560310085471797250', '1560309358598914050', '博士研究生', '博士研究生', 'FRM', 80, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560338934867791874', '0', '定时任务分类', 'JOB_CATEGORY', 'FRM', 16, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560339092900777985', '1560338934867791874', '框架', 'FRM', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560339156134105089', '1560338934867791874', '业务', 'BIZ', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560342111344234497', '0', '定时任务状态', 'JOB_STATUS', 'FRM', 17, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560342186812346370', '1560342111344234497', '运行', 'RUNNING', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1560342250096005121', '1560342111344234497', '停止', 'STOPPED', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1561595062998102017', '0', '三方用户分类', 'THIRD_CATEGORY', 'FRM', 18, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1561595252714860545', '1561595062998102017', '码云GITEE', 'GITEE', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1561595322336112641', '1561595062998102017', '微信WECHAT', 'WECHAT', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1567580351742619650', '0', '系统消息类型', 'MESSAGE_CATEGORY', 'FRM', 19, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1567580424270524418', '1567580351742619650', '系统', 'SYS', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1567580487684206594', '1567580351742619650', '业务', 'BIZ', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1619343323218432002', '0', '移动菜单状态', 'MOBILE_STATUS', 'FRM', 93, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1619343680636047362', '1619343323218432002', '可用', 'ENABLE', 'FRM', 94, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1619343846382358529', '1619343323218432002', '不可用', 'DISABLED', 'FRM', 96, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1619344256295882753', '0', '移动菜单规则', 'MOBILE_REG_TYPE', 'FRM', 97, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1619344428111351809', '1619344256295882753', '正规则', 'YES', 'FRM', 98, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1619344504456073218', '1619344256295882753', '反规则', 'NO', 'FRM', 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1728459107437219842', '0', '系统菜单可见', 'MENU_VISIBLE', 'FRM', 100, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1728459178341928962', '1728459107437219842', '显示', 'TRUE', 'FRM', 101, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1728459493170581505', '1728459107437219842', '隐藏', 'FALSE', 'FRM', 102, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811285429576929282', '0', '轮播图位置', 'DEV_SLIDESHOW_PLACE', 'FRM', 114, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811285517497929730', '1811285429576929282', '后台移动端首页', 'BACK_MOBILE', 'FRM', 115, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811285597999206402', '1811285429576929282', '后台业务首页', 'BACK_INDEX', 'FRM', 116, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811285669692444674', '1811285429576929282', '用户移动端首页', 'CLIENT_MOBILE', 'FRM', 117, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811285778270392321', '1811285429576929282', '后台系统首页', 'BACK_SYS_INDEX', 'FRM', 118, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811286076128890881', '0', '轮播图状态', 'DEV_SLIDESHOW_STATUS', 'FRM', 119, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811286168449716226', '1811286076128890881', '启用', 'ENABLE', 'FRM', 120, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811286273949044737', '1811286076128890881', '禁用', 'DISABLE', 'FRM', 121, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811286542808125441', '0', '轮播图跳转', 'WHETHER_TO_CLICK', 'FRM', 122, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811286611254972418', '1811286542808125441', '启用', 'ENABLE', 'FRM', 123, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811286765760548866', '1811286542808125441', '禁用', 'DISABLE', 'FRM', 124, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811287001631428610', '0', '轮播图跳转方式', 'SKIP_MODE', 'FRM', 125, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811287257836294146', '1811287001631428610', '跳转至URL', 'URL', 'FRM', 126, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811287406557925378', '1811287001631428610', '跳转至路由', 'ROUTER', 'FRM', 127, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811257769085767682', '0', '通知公告类型', 'BIZ_NOTICE_TYPE', 'FRM', 128, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811258110468558849', '1811257769085767682', '通知', 'NOTICE', 'FRM', 129, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811258351498432514', '1811257769085767682', '公告', 'ANNOUNCEMENT', 'FRM', 130, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811258502975721474', '1811257769085767682', '警告', 'WARNING', 'FRM', 131, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811258687793532929', '0', '通知公告状态', 'BIZ_NOTICE_STATUS', 'FRM', 132, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811259114710765570', '1811258687793532929', '已关闭', 'DISABLE', 'FRM', 133, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811259393917194242', '1811258687793532929', '正常', 'ENABLE', 'FRM', 134, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811259821748785154', '0', '通知公告位置', 'BIZ_NOTICE_PLACE', 'FRM', 135, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811260041488371714', '1811259821748785154', '后台移动端首页', 'BACK_MOBILE', 'FRM', 136, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811260212410454018', '1811259821748785154', '后台业务首页', 'BACK_INDEX', 'FRM', 137, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `DEV_DICT` VALUES ('1811260660487950338', '1811259821748785154', '用户移动端首页', 'CLIENT_MOBILE', 'FRM', 138, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543839774776291330', '0', '用户性别类型', 'GENDER', 'nx2UbxB6z6', 'FRM', 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543839901037424642', '1543839774776291330', '男', '男', 'WRrdMORE3c', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543840033980084226', '1543839774776291330', '女', '女', '4MbEORlvBJ', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543860103661809666', '0', '系统菜单类型', 'MENU_TYPE', 'pEnxBN7E2o', 'FRM', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543860239020388354', '1543860103661809666', '目录', 'CATALOG', 'kMKSJSFZgt', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543860305508495361', '1543860103661809666', '菜单', 'MENU', 'FOHDkzOjxa', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543860423485878274', '1543860103661809666', '内链', 'IFRAME', 'u6rH9LmTvz', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1543860477512708098', '1543860103661809666', '外链', 'LINK', 'vOmYaKkhrK', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1544329873407946753', '0', '系统通用状态', 'COMMON_STATUS', 'SpdguT1MTy', 'FRM', 3, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1545397556652027906', '1544329873407946753', '启用', 'ENABLE', '4fPYJApbLH', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1545397556652027907', '1544329873407946753', '停用', 'DISABLED', 'DreeA3e26Y', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547207669861064706', '0', '系统角色分类', 'ROLE_CATEGORY', 'rMCKoVuOoF', 'FRM', 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547207891009937409', '1547207669861064706', '全局', 'GLOBAL', 'gaLjy7wwCw', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547207990075203585', '1547207669861064706', '机构', 'ORG', 'f1B0Zt9PPM', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547227094995705858', '0', '系统机构分类', 'ORG_CATEGORY', '2ATB1ySwLN', 'FRM', 5, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547227670693289985', '1547227094995705858', '部门', 'DEPT', 'XUUCvprRXw', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547228161267474434', '1547227094995705858', '公司', 'COMPANY', '6ntdgLkMOi', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547409689434742786', '0', '系统职位分类', 'POSITION_CATEGORY', 'hPvAP5dxiW', 'FRM', 6, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547409794837602305', '1547409689434742786', '高层', 'HIGH', '7QQxWyMMB0', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547409844779180033', '1547409689434742786', '中层', 'MIDDLE', 'WneMc7T6IT', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547409906406088706', '1547409689434742786', '基层', 'LOW', 'aQhuK5cZ1B', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658113', '0', '用户民族类型', 'NATION', 'OfM9Yo627Z', 'FRM', 7, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658114', '1547641470701658113', '汉族', '汉族', 'IldDHFWUJu', 'FRM', 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658115', '1547641470701658113', '壮族', '壮族', 'BcWVpbu1xR', 'FRM', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658116', '1547641470701658113', '回族', '回族', 'yjVUonNG7d', 'FRM', 3, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658117', '1547641470701658113', '满族', '满族', 'eJi56S3voF', 'FRM', 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658118', '1547641470701658113', '维吾尔族', '维吾尔族', 'EQaAIa8CM3', 'FRM', 5, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658119', '1547641470701658113', '苗族', '苗族', 'jey27sIjJh', 'FRM', 6, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658120', '1547641470701658113', '彝族', '彝族', 'NLeKPrUqmD', 'FRM', 7, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658121', '1547641470701658113', '土家族', '土家族', 'jT8d1Ik5pg', 'FRM', 8, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658122', '1547641470701658113', '藏族', '藏族', 'Hn7hhRzZkK', 'FRM', 9, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658123', '1547641470701658113', '蒙古族', '蒙古族', 'nXOl2haSmn', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658124', '1547641470701658113', '侗族', '侗族', 'sjtNWihEAP', 'FRM', 11, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658125', '1547641470701658113', '布依族', '布依族', 'BUJNxSLxaG', 'FRM', 12, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658126', '1547641470701658113', '瑶族', '瑶族', 'XO7sLv3shh', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658127', '1547641470701658113', '白族', '白族', 'kLKZ70OwP1', 'FRM', 14, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658128', '1547641470701658113', '朝鲜族', '朝鲜族', 'nMRLtjAPfu', 'FRM', 15, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658129', '1547641470701658113', '哈尼族', '哈尼族', 'rAD1IjUe8L', 'FRM', 16, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658130', '1547641470701658113', '黎族', '黎族', 'pgrMlPbuhg', 'FRM', 17, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658131', '1547641470701658113', '哈萨克族', '哈萨克族', 'ZCMR77KXLM', 'FRM', 18, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658132', '1547641470701658113', '傣族', '傣族', 'w7FEqD6b9b', 'FRM', 19, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658133', '1547641470701658113', '畲族', '畲族', 'II3pYCP3gy', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658134', '1547641470701658113', '傈僳族', '傈僳族', 'P4cZRdzfkD', 'FRM', 21, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658135', '1547641470701658113', '东乡族', '东乡族', 'LsDVhLz5aK', 'FRM', 22, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658136', '1547641470701658113', '仡佬族', '仡佬族', 'cRW15z07Wf', 'FRM', 23, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658137', '1547641470701658113', '拉祜族', '拉祜族', 'Rxxpkdtuyi', 'FRM', 24, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658138', '1547641470701658113', '佤族', '佤族', 'BTFWGDBADw', 'FRM', 25, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658139', '1547641470701658113', '水族', '水族', 'rrXze6vNX9', 'FRM', 26, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658140', '1547641470701658113', '纳西族', '纳西族', 'E78hSFKAko', 'FRM', 27, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658141', '1547641470701658113', '羌族', '羌族', 'VoNs4A5D5o', 'FRM', 28, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658142', '1547641470701658113', '土族', '土族', 'UxDL9BKnUm', 'FRM', 29, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658143', '1547641470701658113', '仫佬族', '仫佬族', 'Qn03GasUeA', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658144', '1547641470701658113', '锡伯族', '锡伯族', 'yf4RZRyGzK', 'FRM', 31, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658145', '1547641470701658113', '柯尔克孜族', '柯尔克孜族', 'mCACEtEHGz', 'FRM', 32, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658146', '1547641470701658113', '景颇族', '景颇族', 'XVGhmOIr1v', 'FRM', 33, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658147', '1547641470701658113', '达斡尔族', '达斡尔族', 'pB7plgu56s', 'FRM', 34, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658148', '1547641470701658113', '撒拉族', '撒拉族', '6z5gsbJYwa', 'FRM', 35, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658149', '1547641470701658113', '布朗族', '布朗族', 'qFV4FirD9p', 'FRM', 36, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658150', '1547641470701658113', '毛南族', '毛南族', 'rsNRdCZSMn', 'FRM', 37, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658151', '1547641470701658113', '塔吉克族', '塔吉克族', 'Snn1gGO0Mo', 'FRM', 38, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658152', '1547641470701658113', '普米族', '普米族', 'OfJpLT9wnU', 'FRM', 39, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658153', '1547641470701658113', '阿昌族', '阿昌族', 'GB03mwF8Ou', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658154', '1547641470701658113', '怒族', '怒族', 'kIUgNs9ESr', 'FRM', 41, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658155', '1547641470701658113', '鄂温克族', '鄂温克族', 'TMlFI08Nyp', 'FRM', 42, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658156', '1547641470701658113', '京族', '京族', 'QNeQWqhYjB', 'FRM', 43, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658157', '1547641470701658113', '基诺族', '基诺族', 'LW4sPUhztO', 'FRM', 44, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658158', '1547641470701658113', '德昂族', '德昂族', 'au5pfugMm1', 'FRM', 45, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658159', '1547641470701658113', '保安族', '保安族', 'xAkxr6cORb', 'FRM', 46, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658160', '1547641470701658113', '俄罗斯族', '俄罗斯族', 'XVDaOjSCRC', 'FRM', 47, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658161', '1547641470701658113', '裕固族', '裕固族', 'aDgjM4JVJ4', 'FRM', 48, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658162', '1547641470701658113', '乌孜别克族', '乌孜别克族', 'IPtwpYl2GX', 'FRM', 49, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658163', '1547641470701658113', '门巴族', '门巴族', 'lP5Vim88tt', 'FRM', 50, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658164', '1547641470701658113', '鄂伦春族', '鄂伦春族', 'NQYDgg0vzB', 'FRM', 51, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658165', '1547641470701658113', '独龙族', '独龙族', 'a37PtONVYo', 'FRM', 52, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658166', '1547641470701658113', '赫哲族', '赫哲族', 'JJ97lOiA4D', 'FRM', 53, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658167', '1547641470701658113', '高山族', '高山族', 'zc52WRi5ro', 'FRM', 54, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658168', '1547641470701658113', '珞巴族', '珞巴族', 'v4ZkItC09c', 'FRM', 55, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1547641470701658169', '1547641470701658113', '塔塔尔族 ', '塔塔尔族 ', 'c4AMDTdh2c', 'FRM', 56, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1549019269252104194', '0', '登录设备类型', 'AUTH_DEVICE_TYPE', '29o4YioH2v', 'FRM', 8, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1549019748883349506', '1549019269252104194', 'PC端', 'PC', 'kf2lvOCEcJ', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1549019813924421634', '1549019269252104194', '移动端', 'APP', 'QGygBhRmjE', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1549019864537088002', '1549019269252104194', '小程序', 'MINI', 'uTYA8duLr3', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554678566166323202', '0', '系统字典分类', 'DICT_CATEGORY', 'f4vCzVKlh5', 'FRM', 9, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554678761742524417', '1554678566166323202', '框架', 'FRM', 'ydzbkvUTj4', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554678863617974273', '1554678566166323202', '业务', 'BIZ', 'xI3z1fXU5z', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554679788378120194', '0', '短信发送引擎', 'SMS_ENGINE', 'kPEEms6spU', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554679872054484993', '1554679788378120194', '阿里云', 'ALIYUN', 'GGSVo47jJ0', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554679958398427138', '1554679788378120194', '腾讯云', 'TENCENT', 'jQ4boYgQCK', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554679958398427139', '1554679788378120194', '小诺方舟', 'XIAONUO', 'M3Oie6gy5Z', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554726376265744386', '0', '文件上传引擎', 'FILE_ENGINE', 'Vo3kYSoWu4', 'FRM', 11, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554726639571566593', '1554726376265744386', '本地', 'LOCAL', 'qw9Tr2Dzox', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554726696068841474', '1554726376265744386', '阿里云', 'ALIYUN', 'DdvFFdIFJA', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554726762338844674', '1554726376265744386', '腾讯云', 'TENCENT', 'Ikys8NAfP4', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1554726830844411905', '1554726376265744386', 'MINIO', 'MINIO', 'KXhgDXlijP', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556317797993218049', '0', '邮件发送引擎', 'EMAIL_ENGINE', 'NfnLwwpxlj', 'FRM', 12, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319651447767041', '1556317797993218049', '本地', 'LOCAL', '61rvXPdT9X', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319726962016258', '1556317797993218049', '阿里云', 'ALIYUN', 'DfXhM5Gwau', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166593', '1556317797993218049', '腾讯云', 'TENCENT', 'NRjywm25dm', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166594', '0', '消息推送引擎', 'PUSH_ENGINE', 'qRPg7zAJdP', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166595', '1556319786349166594', '钉钉', 'DINGTALK', 'KDLmGWwMkJ', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166596', '1556319786349166594', '飞书', 'FEISHU', 'Vewyfq4W15', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166597', '1556319786349166594', '企业微信', 'WORKWECHAT', 'JMkb4b45Xd', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166598', '0', '改密验证方式', 'UPDATE_PASSWORD_VALID_TYPE', '9Mqw6ShkbD', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166599', '1556319786349166598', '旧密码', 'OLD', 'pObj0Dl4Xe', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166600', '1556319786349166598', '手机号', 'PHONE', 'WfOLuw9YmO', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166601', '1556319786349166598', '邮箱', 'EMAIL', 'nsGz3lTCqX', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166602', '0', '密码复杂策略', 'PASSWORD_COMPLEXITY_TYPE', 'dCk4dqC6IY', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166603', '1556319786349166602', '无限制', 'REG0', '7SWkXJllso', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166604', '1556319786349166602', '必须包含数字和字母', 'REG1', 'MZm9NDOMlf', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166605', '1556319786349166602', '必须包含数字和大写字母', 'REG2', 'RnOoCgVDv6', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166606', '1556319786349166602', '必须包含数字、大写字母、小写字母和特殊字符', 'REG3', 'q8abKxE74G', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166607', '1556319786349166602', '至少包含数字、字母和特殊字符中的两种', 'REG4', 'SW7iBXX4oY', 'FRM', 50, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166608', '1556319786349166602', '至少包含数字、大写字母、小写字母和特殊字符的三种', 'REG5', 'Fztg672Cva', 'FRM', 60, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166609', '0', '无用户时策略', 'NO_USER_STRATEGY', '49p2f52gYw', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166610', '1556319786349166609', '不允许登录', 'NOT_ALLOW_LOGIN', '5G50YsKslb', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1556319786349166611', '1556319786349166609', '自动创建用户', 'AUTO_CREATE_USER', '7uCMN8LbRL', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1559942543251079169', '0', '系统通用开关', 'COMMON_SWITCH', 'MFJCVVdJgk', 'FRM', 13, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1559942706694717442', '1559942543251079169', '开', 'true', 'kNPE0kSKBO', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1559942777674924034', '1559942543251079169', '关', 'false', 'NNzkxA5SNS', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560306347516461058', '0', '用户证件类型', 'IDCARD_TYPE', '2J7klwyzPJ', 'FRM', 14, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560306502135283714', '1560306347516461058', '身份证', '身份证', 'REJIJwygYr', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560306768913989633', '1560306347516461058', '出生证', '出生证', 'eXyBSq6xwR', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560307009365049346', '1560306347516461058', '军官证', '军官证', 'L4pbqvDHsG', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560307180937248769', '1560306347516461058', '护照', '护照', 's8RAk6LIO3', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309358598914050', '0', '通用文化程度', 'CULTURE_LEVEL', 'pwP5ee8waN', 'FRM', 15, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309494892822530', '1560309358598914050', '小学', '小学', 'oE5AicWtWP', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309602136981505', '1560309358598914050', '中学', '中学', 'DPuwboHfx2', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309700136894465', '1560309358598914050', '高中', '高中', 'u8M3iyacaz', 'FRM', 30, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309744118366209', '1560309358598914050', '中专', '中专', 'CswfpaCuEy', 'FRM', 40, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309783037313026', '1560309358598914050', '大专', '大专', 'hrJGWys2Wj', 'FRM', 50, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309816423972866', '1560309358598914050', '本科', '本科', 'LkR8xvh9tY', 'FRM', 60, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560309855661686785', '1560309358598914050', '硕士研究生', '硕士研究生', '75Vt847ADj', 'FRM', 70, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560310085471797250', '1560309358598914050', '博士研究生', '博士研究生', 'dgOJaPhHzh', 'FRM', 80, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560338934867791874', '0', '定时任务分类', 'JOB_CATEGORY', 'Hodol7eYnE', 'FRM', 16, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560339092900777985', '1560338934867791874', '框架', 'FRM', 'AIBvupunuc', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560339156134105089', '1560338934867791874', '业务', 'BIZ', 'L0kgu7bNYN', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560342111344234497', '0', '定时任务状态', 'JOB_STATUS', 'jlfpcMEVVK', 'FRM', 17, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560342186812346370', '1560342111344234497', '运行', 'RUNNING', 'DHhBaibuBk', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1560342250096005121', '1560342111344234497', '停止', 'STOPPED', 'O6KfK7Cobx', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1561595062998102017', '0', '三方用户分类', 'THIRD_CATEGORY', 'OqMe62ViBY', 'FRM', 18, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1561595252714860545', '1561595062998102017', '码云GITEE', 'GITEE', 'lZq8s2ehA5', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1561595322336112641', '1561595062998102017', '微信WECHAT', 'WECHAT', 'pGhKCHQ7hF', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1567580351742619650', '0', '系统消息类型', 'MESSAGE_CATEGORY', 'UPIN4KJqvT', 'FRM', 19, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1567580424270524418', '1567580351742619650', '系统', 'SYS', 'l7Mc5zE5ib', 'FRM', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1567580487684206594', '1567580351742619650', '业务', 'BIZ', 'nYhpV3ep7j', 'FRM', 20, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1619343323218432002', '0', '移动菜单状态', 'MOBILE_STATUS', 'zhK8QL6NOu', 'FRM', 93, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1619343680636047362', '1619343323218432002', '可用', 'ENABLE', 'pCdVlpHewf', 'FRM', 94, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1619343846382358529', '1619343323218432002', '不可用', 'DISABLED', 'K3FADjJN7G', 'FRM', 96, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1619344256295882753', '0', '移动菜单规则', 'MOBILE_REG_TYPE', '4Lz21CUUTV', 'FRM', 97, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1619344428111351809', '1619344256295882753', '正规则', 'YES', 'Tb4OsxMVgN', 'FRM', 98, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1619344504456073218', '1619344256295882753', '反规则', 'NO', '0ibifiB9mB', 'FRM', 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1728459107437219842', '0', '系统菜单可见', 'MENU_VISIBLE', 'SCF7DIoiOw', 'FRM', 100, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1728459178341928962', '1728459107437219842', '显示', 'TRUE', 'JpeQ3wzeTY', 'FRM', 101, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1728459493170581505', '1728459107437219842', '隐藏', 'FALSE', 's5RGksqfjr', 'FRM', 102, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811257769085767682', '0', '通知公告类型', 'BIZ_NOTICE_TYPE', 'iXHgzfK7W5', 'FRM', 128, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811258110468558849', '1811257769085767682', '通知', 'NOTICE', 'rNtBUWLnag', 'FRM', 129, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811258351498432514', '1811257769085767682', '公告', 'ANNOUNCEMENT', '578Xq2zTkK', 'FRM', 130, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811258502975721474', '1811257769085767682', '警告', 'WARNING', 'cVjlhwLiE0', 'FRM', 131, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811258687793532929', '0', '通知公告状态', 'BIZ_NOTICE_STATUS', '1OcT1FGyRU', 'FRM', 132, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811259114710765570', '1811258687793532929', '已关闭', 'DISABLE', 'SB07WaNLrC', 'FRM', 133, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811259393917194242', '1811258687793532929', '正常', 'ENABLE', 'O27ZBLXOF5', 'FRM', 134, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811259821748785154', '0', '通知公告位置', 'BIZ_NOTICE_PLACE', '0gnQn6GnJs', 'FRM', 135, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811260041488371714', '1811259821748785154', '后台移动端首页', 'BACK_MOBILE', 'g7E1fXh3l8', 'FRM', 136, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811260212410454018', '1811259821748785154', '后台业务首页', 'BACK_INDEX', 'GP4eqroglk', 'FRM', 137, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811260660487950338', '1811259821748785154', '用户移动端首页', 'CLIENT_MOBILE', 'gnDATqoDpO', 'FRM', 138, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811285429576929282', '0', '轮播图位置', 'DEV_SLIDESHOW_PLACE', '1HQeP9ZwrY', 'FRM', 114, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811285517497929730', '1811285429576929282', '后台移动端首页', 'BACK_MOBILE', 'Lcd7Oc3SMD', 'FRM', 115, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811285597999206402', '1811285429576929282', '后台业务首页', 'BACK_INDEX', 'J1utDwWvGZ', 'FRM', 116, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811285669692444674', '1811285429576929282', '用户移动端首页', 'CLIENT_MOBILE', 'QOA6Gh75nr', 'FRM', 117, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811285778270392321', '1811285429576929282', '后台系统首页', 'BACK_SYS_INDEX', 'vdeOwzBXBr', 'FRM', 118, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811286076128890881', '0', '轮播图状态', 'DEV_SLIDESHOW_STATUS', 'T5lIZkVfTj', 'FRM', 119, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811286168449716226', '1811286076128890881', '启用', 'ENABLE', 'MuZagpmxPv', 'FRM', 120, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811286273949044737', '1811286076128890881', '禁用', 'DISABLE', '5gpxaee739', 'FRM', 121, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811286542808125441', '0', '轮播图跳转', 'WHETHER_TO_CLICK', 'yBB2WcLn2V', 'FRM', 122, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811286611254972418', '1811286542808125441', '启用', 'ENABLE', 'ac87rrN9WF', 'FRM', 123, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811286765760548866', '1811286542808125441', '禁用', 'DISABLE', 'la3FazhpXJ', 'FRM', 124, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811287001631428610', '0', '轮播图跳转方式', 'SKIP_MODE', 'PGtlHTRTEA', 'FRM', 125, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811287257836294146', '1811287001631428610', '跳转至URL', 'URL', 'xmKzrPyyrM', 'FRM', 126, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `DEV_DICT` VALUES ('1811287406557925378', '1811287001631428610', '跳转至路由', 'ROUTER', 'lrtd0Zt0Di', 'FRM', 127, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -- ---------------------------- -- Table structure for DEV_EMAIL @@ -389,7 +565,7 @@ CREATE TABLE `DEV_EMAIL` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '邮件' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '邮件' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_EMAIL @@ -419,7 +595,7 @@ CREATE TABLE `DEV_FILE` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '文件' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '文件' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_FILE @@ -445,7 +621,7 @@ CREATE TABLE `DEV_JOB` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '定时任务' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '定时任务' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_JOB @@ -480,7 +656,7 @@ CREATE TABLE `DEV_LOG` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '日志' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '日志' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_LOG @@ -502,12 +678,36 @@ CREATE TABLE `DEV_MESSAGE` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '站内信' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '站内信' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_MESSAGE -- ---------------------------- +-- ---------------------------- +-- Table structure for DEV_PUSH +-- ---------------------------- +DROP TABLE IF EXISTS `DEV_PUSH`; +CREATE TABLE `DEV_PUSH` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL, + `ENGINE` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '消息引擎', + `TYPE` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '消息类别', + `TITLE` text CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '消息标题', + `CONTENT` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '消息内容', + `RECEIPT_INFO` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '回执信息', + `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '消息推送' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of DEV_PUSH +-- ---------------------------- + -- ---------------------------- -- Table structure for DEV_RELATION -- ---------------------------- @@ -519,12 +719,37 @@ CREATE TABLE `DEV_RELATION` ( `CATEGORY` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '分类', `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '关系' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '关系' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_RELATION -- ---------------------------- +-- ---------------------------- +-- Table structure for DEV_SLIDESHOW +-- ---------------------------- +DROP TABLE IF EXISTS `DEV_SLIDESHOW`; +CREATE TABLE `DEV_SLIDESHOW` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '主键', + `TITLE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '标题', + `PLACE` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '展示位置', + `IMAGE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '图片', + `PATH_DETAILS` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '路径详情', + `STATUS` varchar(10) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '状态', + `SORT_CODE` int(11) NULL DEFAULT NULL COMMENT '排序', + `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '更新时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '更新用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '轮播图' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of DEV_SLIDESHOW +-- ---------------------------- + -- ---------------------------- -- Table structure for DEV_SMS -- ---------------------------- @@ -544,12 +769,31 @@ CREATE TABLE `DEV_SMS` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '短信' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '短信' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of DEV_SMS -- ---------------------------- +-- ---------------------------- +-- Table structure for DEV_WEAK_PASSWORD +-- ---------------------------- +DROP TABLE IF EXISTS `DEV_WEAK_PASSWORD`; +CREATE TABLE `DEV_WEAK_PASSWORD` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL, + `PASSWORD` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '弱密码', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '弱密码库' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of DEV_WEAK_PASSWORD +-- ---------------------------- + -- ---------------------------- -- Table structure for GEN_BASIC -- ---------------------------- @@ -579,7 +823,7 @@ CREATE TABLE `GEN_BASIC` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '生成基础' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '生成基础' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of GEN_BASIC @@ -612,7 +856,7 @@ CREATE TABLE `GEN_CONFIG` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '生成配置' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '生成配置' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of GEN_CONFIG @@ -643,16 +887,16 @@ CREATE TABLE `MOBILE_RESOURCE` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '移动资源' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '移动资源' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of MOBILE_RESOURCE -- ---------------------------- INSERT INTO `MOBILE_RESOURCE` VALUES ('1623380023993298945', NULL, '业务', '217gcp9ifi', 'MODULE', NULL, NULL, NULL, 'container-outlined', '#1890ff', NULL, NULL, 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `MOBILE_RESOURCE` VALUES ('1623380258656219138', '0', '机构管理', NULL, 'MENU', '1623380023993298945', 'MENU', '/pages/biz/org/index', 'apartment-outlined', '#1890ff', 'YES', 'ENABLE', 0, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `MOBILE_RESOURCE` VALUES ('1623380614295449601', '0', '岗位管理', NULL, 'MENU', '1623380023993298945', 'MENU', '/pages/biz/position/index', 'robot-outlined', '#9c28b1', 'YES', 'ENABLE', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `MOBILE_RESOURCE` VALUES ('1623380765202313218', '0', '人员管理', NULL, 'MENU', '1623380023993298945', 'MENU', '/pages/biz/user/index', 'team-outlined', '#fed835', 'YES', 'ENABLE', 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `MOBILE_RESOURCE` VALUES ('1623381127095250946', '0', '更多', NULL, 'MENU', '1623380023993298945', 'CATALOG', '7029146815941316608', 'small-dash-outlined', '#f1627e', 'YES', 'ENABLE', 7, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `MOBILE_RESOURCE` VALUES ('1623380258656219138', '0', '机构管理', 'UJ3Iwy3jsW', 'MENU', '1623380023993298945', 'MENU', '/pages/biz/org/index', 'apartment-outlined', '#1890ff', 'YES', 'ENABLE', 0, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `MOBILE_RESOURCE` VALUES ('1623380614295449601', '0', '岗位管理', 'sjIY9oGYir', 'MENU', '1623380023993298945', 'MENU', '/pages/biz/position/index', 'robot-outlined', '#9c28b1', 'YES', 'ENABLE', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `MOBILE_RESOURCE` VALUES ('1623380765202313218', '0', '人员管理', 'sjIY9oGYir', 'MENU', '1623380023993298945', 'MENU', '/pages/biz/user/index', 'team-outlined', '#fed835', 'YES', 'ENABLE', 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `MOBILE_RESOURCE` VALUES ('1623381127095250946', '0', '更多', 'mFVJNzE7gx', 'MENU', '1623380023993298945', 'CATALOG', '7029146815941316608', 'small-dash-outlined', '#f1627e', 'YES', 'ENABLE', 7, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `MOBILE_RESOURCE` VALUES ('1623381298801668098', '1623380258656219138', '新增机构', 'mobileBizOrgAdd', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `MOBILE_RESOURCE` VALUES ('1623381632131395586', '1623380258656219138', '编辑机构', 'mobileBizOrgEdit', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `MOBILE_RESOURCE` VALUES ('1623381698837606401', '1623380258656219138', '删除机构', 'mobileBizOrgDelete', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 3, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); @@ -668,17 +912,17 @@ INSERT INTO `MOBILE_RESOURCE` VALUES ('1623696346547859458', '162338076520231321 -- ---------------------------- DROP TABLE IF EXISTS `SYS_GROUP`; CREATE TABLE `SYS_GROUP` ( - `ID` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '主键', - `NAME` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '名称', - `REMARK` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '备注', - `SORT_CODE` int(11) NULL DEFAULT NULL COMMENT '排序码', - `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', - `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', - `CREATE_TIME` datetime(0) NULL DEFAULT NULL COMMENT '创建时间', - `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', - `UPDATE_TIME` datetime(0) NULL DEFAULT NULL COMMENT '修改时间', - `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', - PRIMARY KEY (`ID`) USING BTREE + `ID` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '主键', + `NAME` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '名称', + `REMARK` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '备注', + `SORT_CODE` int(11) NULL DEFAULT NULL COMMENT '排序码', + `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE ) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '用户组' ROW_FORMAT = Dynamic; -- ---------------------------- @@ -704,7 +948,7 @@ CREATE TABLE `SYS_ORG` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '组织' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '组织' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of SYS_ORG @@ -720,6 +964,26 @@ INSERT INTO `SYS_ORG` VALUES ('1543842934270394376', '1543842934270394368', '154 INSERT INTO `SYS_ORG` VALUES ('1543842934270394377', '1543842934270394368', '1543837863788879912', '计划营销部', 'w742mipwer', 'DEPT', 10, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_ORG` VALUES ('1543842934270394378', '1543842934270394368', '1543837863788879917', '后勤保卫部', 'b71pvf46do', 'DEPT', 11, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +-- ---------------------------- +-- Table structure for SYS_ORG_EXT +-- ---------------------------- +DROP TABLE IF EXISTS `SYS_ORG_EXT`; +CREATE TABLE `SYS_ORG_EXT` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL, + `ORG_ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '组织id', + `SOURCE_FROM_TYPE` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '来源类别', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '组织扩展' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of SYS_ORG_EXT +-- ---------------------------- + -- ---------------------------- -- Table structure for SYS_POSITION -- ---------------------------- @@ -738,7 +1002,7 @@ CREATE TABLE `SYS_POSITION` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '职位' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '职位' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of SYS_POSITION @@ -760,7 +1024,7 @@ CREATE TABLE `SYS_RELATION` ( `CATEGORY` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '分类', `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '关系' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '关系' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of SYS_RELATION @@ -900,9 +1164,9 @@ INSERT INTO `SYS_RELATION` VALUES ('1728455598297944066', '1570687866138206208', INSERT INTO `SYS_RELATION` VALUES ('1728455598297944067', '1570687866138206208', '1548901111999773978', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1548901111999773978\",\"buttonInfo\":[\"1571129529564758017\",\"1571129929961406466\",\"1571130756155408386\",\"1571130811058847745\"]}'); INSERT INTO `SYS_RELATION` VALUES ('1728455598297944068', '1570687866138206208', '1548901111999773979', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1548901111999773979\",\"buttonInfo\":[\"1571130973294526465\",\"1571131043532341249\",\"1571131137006600193\",\"1571131427361488898\",\"1571131544973967361\",\"1571131727656878081\",\"1571132076853657601\",\"1635110416263262209\",\"1635110536451043329\"]}'); INSERT INTO `SYS_RELATION` VALUES ('1728455598297944069', '1570687866138206208', '1548901111999773980', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1548901111999773980\",\"buttonInfo\":[\"1571132393993371649\",\"1571132468178026497\",\"1571132576143605761\",\"1571132658851086338\"]}'); - -INSERT INTO `SYS_RELATION` VALUES ('1812126273234231297', '1570687866138206208', '1811290937444667393', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1811290937444667393\",\"buttonInfo\":[\"1811290937469833217\",\"1811290937486610434\",\"1811290937494999042\",\"1811290937503387650\",\"1811328402989662210\",\"1811330359695400961\"]}'); INSERT INTO `SYS_RELATION` VALUES ('1728521685882490892', '1570687866138206208', '1811800584425500674', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1811800584425500674\",\"buttonInfo\":[]}'); +INSERT INTO `SYS_RELATION` VALUES ('1812126273234231297', '1570687866138206208', '1811290937444667393', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1811290937444667393\",\"buttonInfo\":[\"1811290937469833217\",\"1811290937486610434\",\"1811290937494999042\",\"1811290937503387650\",\"1811328402989662210\",\"1811330359695400961\"]}'); +INSERT INTO `SYS_RELATION` VALUES ('1813959903803842580', '1570687866138206208', '1813959658013433858', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1813959658013433858\",\"buttonInfo\":[]}'); INSERT INTO `SYS_RELATION` VALUES ('1813960110876631044', '1570687866138206208', '/biz/notice/add', 'SYS_ROLE_HAS_PERMISSION', '{\"apiUrl\":\"/biz/notice/add\",\"scopeCategory\":\"SCOPE_ALL\",\"scopeDefineOrgIdList\":[]}'); INSERT INTO `SYS_RELATION` VALUES ('1813960110876631045', '1570687866138206208', '/biz/notice/delete', 'SYS_ROLE_HAS_PERMISSION', '{\"apiUrl\":\"/biz/notice/delete\",\"scopeCategory\":\"SCOPE_ALL\",\"scopeDefineOrgIdList\":[]}'); INSERT INTO `SYS_RELATION` VALUES ('1813960110876631046', '1570687866138206208', '/biz/notice/detail', 'SYS_ROLE_HAS_PERMISSION', '{\"apiUrl\":\"/biz/notice/detail\",\"scopeCategory\":\"SCOPE_ALL\",\"scopeDefineOrgIdList\":[]}'); @@ -910,10 +1174,9 @@ INSERT INTO `SYS_RELATION` VALUES ('1813960110876631047', '1570687866138206208', INSERT INTO `SYS_RELATION` VALUES ('1813960110876631048', '1570687866138206208', '/biz/notice/edit', 'SYS_ROLE_HAS_PERMISSION', '{\"apiUrl\":\"/biz/notice/edit\",\"scopeCategory\":\"SCOPE_ALL\",\"scopeDefineOrgIdList\":[]}'); INSERT INTO `SYS_RELATION` VALUES ('1813960110876631049', '1570687866138206208', '/biz/notice/enableStatus', 'SYS_ROLE_HAS_PERMISSION', '{\"apiUrl\":\"/biz/notice/enableStatus\",\"scopeCategory\":\"SCOPE_ALL\",\"scopeDefineOrgIdList\":[]}'); INSERT INTO `SYS_RELATION` VALUES ('1813960110876631050', '1570687866138206208', '/biz/notice/page', 'SYS_ROLE_HAS_PERMISSION', '{\"apiUrl\":\"/biz/notice/page\",\"scopeCategory\":\"SCOPE_ALL\",\"scopeDefineOrgIdList\":[]}'); -INSERT INTO `SYS_RELATION` VALUES ('1813959903803842580', '1570687866138206208', '1813959658013433858', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1813959658013433858\",\"buttonInfo\":[]}'); - INSERT INTO `SYS_RELATION` VALUES ('1871280434097565701', '1570687866138206208', '1870158678418993154', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1870158678418993154\",\"buttonInfo\":[\"1870158678481907713\",\"1870158678481907714\",\"1870158678481907715\",\"1870158678481907716\"]}'); INSERT INTO `SYS_RELATION` VALUES ('1871280434097565751', '1570687866138206208', '1871278073018986498', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1871278073018986498\",\"buttonInfo\":[\"1871278073086095361\",\"1871278073086095362\",\"1871278073086095363\",\"1871278073086095364\",\"1871280381043814402\"]}'); +INSERT INTO `SYS_RELATION` VALUES ('1909233948956041233', '1570687866138206208', '1548901111999771927', 'SYS_ROLE_HAS_RESOURCE', '{\"menuId\":\"1548901111999771927\",\"buttonInfo\":[]}'); -- ---------------------------- -- Table structure for SYS_RESOURCE @@ -941,7 +1204,7 @@ CREATE TABLE `SYS_RESOURCE` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '资源' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '资源' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of SYS_RESOURCE @@ -960,6 +1223,7 @@ INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999771626', '0', '基础工具', N INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999771726', '1548901111999771626', '文件管理', 'devFile', 'n25k83x4sy', 'MENU', '1548901111999770525', 'MENU', '/dev/file/index', 'dev/file/index', 'copy-outlined', NULL, NULL, 14, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999771826', '1548901111999771626', '邮件推送', 'devEmail', 'x4fx2a91tq', 'MENU', '1548901111999770525', 'MENU', '/dev/email/index', 'dev/email/index', 'send-outlined', NULL, NULL, 15, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999771926', '1548901111999771626', '短信发送', 'devSms', 'nnjsr7tkrs', 'MENU', '1548901111999770525', 'MENU', '/dev/sms/index', 'dev/sms/index', 'mail-outlined', NULL, NULL, 16, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999771927', '1548901111999771626', '消息推送', 'devPush', 'ECKjtnzE9A', 'MENU', '1548901111999770525', 'MENU', '/dev/push/index', 'dev/push/index', 'appstore-outlined', NULL, NULL, 16, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999772026', '1548901111999771626', '站内信息', 'devMessage', '0yitzu8786', 'MENU', '1548901111999770525', 'MENU', '/dev/message/index', 'dev/message/index', 'message-outlined', NULL, NULL, 17, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999772126', '0', '系统运维', NULL, '3poiqgf7zx', 'MENU', '1548901111999770525', 'CATALOG', '/a0l7fxfq3m', NULL, 'hdd-outlined', NULL, NULL, 18, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1548901111999772226', '1548901111999772126', '三方用户', 'authThird', 'xf89fmzrtz', 'MENU', '1548901111999770525', 'MENU', '/auth/third', 'auth/third/index', 'team-outlined', NULL, NULL, 19, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); @@ -1022,11 +1286,8 @@ INSERT INTO `SYS_RESOURCE` VALUES ('1635110416263262209', '1548901111999773979', INSERT INTO `SYS_RESOURCE` VALUES ('1635110536451043329', '1548901111999773979', '批量导出', NULL, 'bizUserBatchExport', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 9, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1689891405367353346', '0', '业务字典', 'bizDict', NULL, 'MENU', '1548901111999773976', 'MENU', '/biz/dict/index', 'biz/dict/index', 'read-outlined', NULL, NULL, 12, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1689892202679377921', '1689891405367353346', '编辑字典', NULL, 'bizDictEdit', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `SYS_RESOURCE` VALUES ('1689894316554067969', '1548901111999773426', '高德地图', 'gaodeMap', NULL, 'MENU', '1548901111999770525', 'MENU', '/exm/map/gaodeMap', 'exm/map/gaodeMap', 'environment-outlined', NULL, NULL, 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `SYS_RESOURCE` VALUES ('1689894577238450177', '1548901111999773426', '百度地图', 'baiduMap', NULL, 'MENU', '1548901111999770525', 'MENU', '/exm/map/baiduMap', 'exm/map/baiduMap', 'compass-outlined', NULL, NULL, 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); - -INSERT INTO `SYS_RESOURCE` VALUES ('1811800584425500674', '1548901111999772126', '轮播图', 'slideshow', 'XPLrleqZof', 'MENU', '1548901111999770525', 'MENU', '/dev/slideshow', 'dev/slideshow/index', 'file-image-outlined', NULL, 'TRUE', 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -INSERT INTO `SYS_RESOURCE` VALUES ('1813959658013433858', '0', '业务首页', 'bizIndex', NULL, 'MENU', '1548901111999773976', 'MENU', '/biz/index', 'biz/index/index', 'home-outlined', NULL, 'TRUE', 0, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `SYS_RESOURCE` VALUES ('1689894316554067969', '1548901111999773426', '高德地图', 'gaodeMap', 'noten', 'MENU', '1548901111999770525', 'MENU', '/exm/map/gaodeMap', 'exm/map/gaodeMap', 'environment-outlined', NULL, NULL, 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `SYS_RESOURCE` VALUES ('1689894577238450177', '1548901111999773426', '百度地图', 'baiduMap', 'noten', 'MENU', '1548901111999770525', 'MENU', '/exm/map/baiduMap', 'exm/map/baiduMap', 'compass-outlined', NULL, NULL, 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1811290937444667393', '0', '通知公告', 'notice', 'oFordpxYEi', 'MENU', '1548901111999773976', 'MENU', '/biz/notice', 'biz/notice/index', 'appstore-outlined', NULL, NULL, 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1811290937469833217', '1811290937444667393', '新增通知公告', NULL, 'bizNoticeAdd', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1811290937486610434', '1811290937444667393', '编辑通知公告', NULL, 'bizNoticeEdit', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); @@ -1034,7 +1295,8 @@ INSERT INTO `SYS_RESOURCE` VALUES ('1811290937494999042', '1811290937444667393', INSERT INTO `SYS_RESOURCE` VALUES ('1811290937503387650', '1811290937444667393', '批量删除', NULL, 'bizNoticeBatchDelete', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 4, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1811328402989662210', '1811290937444667393', '更新状态', NULL, 'bizNoticerUpdateStatus', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 5, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1811330359695400961', '1811290937444667393', '通知公告详情', NULL, 'bizNoticeDetail', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 6, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); - +INSERT INTO `SYS_RESOURCE` VALUES ('1811800584425500674', '1548901111999772126', '轮播图', 'slideshow', 'XPLrleqZof', 'MENU', '1548901111999770525', 'MENU', '/dev/slideshow', 'dev/slideshow/index', 'file-image-outlined', NULL, 'TRUE', 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); +INSERT INTO `SYS_RESOURCE` VALUES ('1813959658013433858', '0', '业务首页', 'bizIndex', 'vY0Qc3fkcR', 'MENU', '1548901111999773976', 'MENU', '/biz/index', 'biz/index/index', 'home-outlined', NULL, 'TRUE', 0, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1870158678418993154', '1548901111999770726', '用户组管理', 'groupIndex', 'd4KUq2ZnK7', 'MENU', '1548901111999770525', 'MENU', '/sys/group', 'sys/group/index', 'team-outlined', NULL, 'TRUE', 99, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1871278073018986498', '1548901111999773977', '用户组', 'bizGroupIndex', 'HQVofyRf7Z', 'MENU', '1548901111999773976', 'MENU', '/biz/group', 'biz/group/index', 'team-outlined', NULL, 'TRUE', 55, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_RESOURCE` VALUES ('1871278073086095361', '1871278073018986498', '新增用户组', NULL, 'bizGroupAdd', 'BUTTON', NULL, NULL, NULL, NULL, NULL, NULL, NULL, 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); @@ -1061,7 +1323,7 @@ CREATE TABLE `SYS_ROLE` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '角色' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '角色' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of SYS_ROLE @@ -1127,7 +1389,7 @@ CREATE TABLE `SYS_USER` ( `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '用户' ROW_FORMAT = DYNAMIC; +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '用户' ROW_FORMAT = Dynamic; -- ---------------------------- -- Records of SYS_USER @@ -1135,43 +1397,45 @@ CREATE TABLE `SYS_USER` ( INSERT INTO `SYS_USER` VALUES ('1543837863788879871', '', NULL, 'superAdmin', '207cf410532f92a47dee245ce9b11ff71f578ebd763eb3bbea44ebd043d018fb', '超管', NULL, '男', NULL, '1990-01-01', '汉', '', '', '', '身份证', '', NULL, NULL, NULL, NULL, NULL, NULL, 'eb77186abe605f8de5958df60ef4a279', 'superAdmin@foxmail.com', NULL, NULL, '', '', '', '0000', '2022-07-04', '1543842934270394368', '1543899639134019584', 'C1', NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, 'ENABLE', 1, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); INSERT INTO `SYS_USER` VALUES ('1543837863788879873', '', NULL, 'bizAdmin', '207cf410532f92a47dee245ce9b11ff71f578ebd763eb3bbea44ebd043d018fb', '业管', NULL, '男', NULL, '1995-01-01', '汉', '', '', '', '身份证', '', NULL, NULL, NULL, NULL, NULL, NULL, '9c8f683ccff14071f90f1f51ba83f069', 'bizAdmin@foxmail.com', NULL, NULL, '', '', '', '0001', '2022-07-04', '1543842934270394368', '1543899639134019584', 'C1', NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, 'ENABLE', 2, NULL, 'NOT_DELETE', NULL, NULL, NULL, NULL); -DROP TABLE IF EXISTS `BIZ_NOTICE`; -CREATE TABLE `BIZ_NOTICE` ( - `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '主键', - `TITLE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '标题', - `IMAGE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '封面图', - `CONTENT` text CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '内容', - `DIGEST` varchar(300) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '摘要', - `TYPE` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '类型', - `PLACE` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '发布位置', - `STATUS` varchar(10) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '状态', - `SORT_CODE` int(10) NULL DEFAULT NULL COMMENT '排序', - `REMARK` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '备注', - `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', - `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', - `CREATE_TIME` datetime(0) NULL DEFAULT NULL COMMENT '创建时间', - `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', - `UPDATE_TIME` datetime(0) NULL DEFAULT NULL COMMENT '更新时间', - `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '更新用户', - PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '通知公告' ROW_FORMAT = Dynamic; +-- ---------------------------- +-- Table structure for SYS_USER_EXT +-- ---------------------------- +DROP TABLE IF EXISTS `SYS_USER_EXT`; +CREATE TABLE `SYS_USER_EXT` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL, + `USER_ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '用户id', + `SOURCE_FROM_TYPE` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '来源类别', + `PASSWORD_UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '密码修改日期', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '用户扩展' ROW_FORMAT = Dynamic; -DROP TABLE IF EXISTS `DEV_SLIDESHOW`; -CREATE TABLE `DEV_SLIDESHOW` ( - `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '主键', - `TITLE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '标题', - `PLACE` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '展示位置', - `IMAGE` varchar(200) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '图片', - `PATH_DETAILS` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '路径详情', - `STATUS` varchar(10) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '状态', - `SORT_CODE` int(255) NULL DEFAULT NULL COMMENT '排序', - `EXT_JSON` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '扩展信息', - `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', - `CREATE_TIME` datetime(0) NULL DEFAULT NULL COMMENT '创建时间', - `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', - `UPDATE_TIME` datetime(0) NULL DEFAULT NULL COMMENT '更新时间', - `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '更新用户', - PRIMARY KEY (`ID`) USING BTREE -) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '轮播图' ROW_FORMAT = Dynamic; +-- ---------------------------- +-- Records of SYS_USER_EXT +-- ---------------------------- + +-- ---------------------------- +-- Table structure for SYS_USER_PASSWORD +-- ---------------------------- +DROP TABLE IF EXISTS `SYS_USER_PASSWORD`; +CREATE TABLE `SYS_USER_PASSWORD` ( + `ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL, + `USER_ID` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '用户id', + `PASSWORD` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '密码', + `DELETE_FLAG` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '删除标志', + `CREATE_TIME` datetime NULL DEFAULT NULL COMMENT '创建时间', + `CREATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建用户', + `UPDATE_TIME` datetime NULL DEFAULT NULL COMMENT '修改时间', + `UPDATE_USER` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '修改用户', + PRIMARY KEY (`ID`) USING BTREE +) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '用户密码' ROW_FORMAT = Dynamic; + +-- ---------------------------- +-- Records of SYS_USER_PASSWORD +-- ---------------------------- SET FOREIGN_KEY_CHECKS = 1; diff --git a/snowy-web-app/src/main/resources/_sql/数据库说明.md b/snowy-web-app/src/main/resources/_sql/数据库说明.md deleted file mode 100644 index fd1afec8..00000000 --- a/snowy-web-app/src/main/resources/_sql/数据库说明.md +++ /dev/null @@ -1,6 +0,0 @@ -# 数据库说明 -注意:sql文件为mysql数据库的文件,如需其他数据库文件,可通过Navicat数据传输等工具传输到目标数据库。 - -文档地址:https://xiaonuo.vip/doc - -如果有其他问题请在群内提出。 \ No newline at end of file diff --git a/snowy-web-app/src/main/resources/application.properties b/snowy-web-app/src/main/resources/application.properties new file mode 100644 index 00000000..7341fe54 --- /dev/null +++ b/snowy-web-app/src/main/resources/application.properties @@ -0,0 +1,209 @@ +######################################### +# server configuration +######################################### +server.port=82 + +######################################### +# spring allow-circular-references +######################################### +spring.main.allow-circular-references=true + +######################################### +# spring profiles configuration +######################################### +spring.profiles.active=local +#spring.profiles.active=test +#spring.profiles.active=prod + +######################################### +# multipart configuration +######################################### +spring.servlet.multipart.max-request-size=100MB +spring.servlet.multipart.max-file-size=100MB + +######################################### +# datasource configuration +######################################### + +# mysql +spring.datasource.dynamic.datasource.master.driver-class-name=com.mysql.cj.jdbc.Driver +spring.datasource.dynamic.datasource.master.url=jdbc:mysql://localhost:3306/snowy?useUnicode=true&characterEncoding=utf-8&useSSL=false&allowPublicKeyRetrieval=true&nullCatalogMeansCurrent=true&useInformationSchema=true&rewriteBatchedStatements=true +spring.datasource.dynamic.datasource.master.username=root +spring.datasource.dynamic.datasource.master.password=12345678 +spring.datasource.dynamic.strict=true + +# postgres +#spring.datasource.dynamic.datasource.master.driver-class-name=org.postgresql.Driver +#spring.datasource.dynamic.datasource.master.url=jdbc:postgresql://localhost:5432/snowy +#spring.datasource.dynamic.datasource.master.username=postgres +#spring.datasource.dynamic.datasource.master.password=123456 +#spring.datasource.dynamic.strict=true + +# oracle +#spring.datasource.dynamic.datasource.master.driver-class-name=oracle.jdbc.OracleDriver +#spring.datasource.dynamic.datasource.master.url=jdbc:oracle:thin:@//127.0.0.1:1521/ORCL?remarksReporting=true +#spring.datasource.dynamic.datasource.master.username=SNOWY +#spring.datasource.dynamic.datasource.master.password=12345678 +#spring.datasource.dynamic.strict=true + +# mssql +#spring.datasource.dynamic.datasource.master.driver-class-name=com.microsoft.sqlserver.jdbc.SQLServerDriver +#spring.datasource.dynamic.datasource.master.url=jdbc:sqlserver://localhost:1433;DatabaseName=SNOWY +#spring.datasource.dynamic.datasource.master.username=sa +#spring.datasource.dynamic.datasource.master.password=123456 +#spring.datasource.dynamic.strict=true + +# dm database +#spring.datasource.dynamic.datasource.master.driver-class-name=dm.jdbc.driver.DmDriver +#spring.datasource.dynamic.datasource.master.url=jdbc:dm://localhost:5236/SNOWY +#spring.datasource.dynamic.datasource.master.username=SYSDBA +#spring.datasource.dynamic.datasource.master.password=SYSDBA +#spring.datasource.dynamic.strict=true + +# kingbase database +#spring.datasource.dynamic.datasource.master.driver-class-name=com.kingbase8.Driver +#spring.datasource.dynamic.datasource.master.url=jdbc:kingbase8://localhost:54321/snowy +#spring.datasource.dynamic.datasource.master.username=SYSTEM +#spring.datasource.dynamic.datasource.master.password=123456 +#spring.datasource.dynamic.strict=true + +# druid monitor configuration +spring.datasource.druid.stat-view-servlet.enabled=true +spring.datasource.druid.stat-view-servlet.login-username=admin +spring.datasource.druid.stat-view-servlet.login-password=123456 + +# druid global configuration +spring.datasource.dynamic.public-key=MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAMWiTVtdXFVrgFHDDKELZM0SywkWY3KjugN90eY5Sogon1j8Y0ClPF7nx3FuE7pAeBKiv7ChIS0vvx/59WUpKmUCAwEAAQ== +spring.datasource.dynamic.druid.initial-size=5 +spring.datasource.dynamic.druid.max-active=20 +spring.datasource.dynamic.druid.min-idle=5 +spring.datasource.dynamic.druid.max-wait=60000 +spring.datasource.dynamic.druid.pool-prepared-statements=true +spring.datasource.dynamic.druid.max-pool-prepared-statement-per-connection-size=20 +spring.datasource.dynamic.druid.validation-query-timeout=2000 +spring.datasource.dynamic.druid.test-on-borrow=false +spring.datasource.dynamic.druid.test-on-return=false +spring.datasource.dynamic.druid.test-while-idle=true +spring.datasource.dynamic.druid.time-between-eviction-runs-millis=6000 +spring.datasource.dynamic.druid.min-evictable-idle-time-millis=300000 +spring.datasource.dynamic.druid.filters=stat +spring.datasource.dynamic.druid.break-after-acquire-failure=false + +######################################### +# jackson configuration +######################################### +spring.jackson.time-zone=GMT+8 +spring.jackson.date-format=yyyy-MM-dd HH:mm:ss +spring.jackson.locale=zh_CN +spring.jackson.serialization.write-dates-as-timestamps=false +######################################### +# redis configuration +######################################### +spring.data.redis.database=1 +spring.data.redis.host=127.0.0.1 +spring.data.redis.port=6379 +spring.data.redis.password= +spring.data.redis.timeout=10s + +spring.data.redis.lettuce.pool.max-active=200 +spring.data.redis.lettuce.pool.max-wait=-1ms +spring.data.redis.lettuce.pool.max-idle=10 +spring.data.redis.lettuce.pool.min-idle=0 + +######################################### +# mybatis-plus configuration +######################################### +mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl +mybatis-plus.configuration.jdbc-type-for-null=null +mybatis-plus.global-config.banner=false +mybatis-plus.global-config.enable-sql-runner=true +mybatis-plus.global-config.db-config.id-type=ASSIGN_ID +mybatis-plus.global-config.db-config.logic-delete-field=DELETE_FLAG +mybatis-plus.global-config.db-config.logic-delete-value=DELETED +mybatis-plus.global-config.db-config.logic-not-delete-value=NOT_DELETE +mybatis-plus.mapper-locations=classpath*:vip/xiaonuo/**/mapping/*.xml,com/bstek/**/mapping/*.xml +mybatis-plus.type-handlers-package=vip.xiaonuo.common.handler + +######################################### +# easy-trans configuration +######################################### +easy-trans.is-enable-redis=true +easy-trans.is-enable-global=true +easy-trans.is-enable-tile=true +easy-trans.is-enable-cloud=false + +######################################### +# sa-token configuration +######################################### +sa-token.token-name=token +sa-token.timeout=2592000 +sa-token.active-timeout=-1 +sa-token.is-concurrent=true +sa-token.is-share=false +sa-token.max-login-count=-1 +sa-token.token-style=random-32 +sa-token.is-log=false +sa-token.is-print=false + +# sa-token alone-redis configuration +sa-token.alone-redis.database=2 +sa-token.alone-redis.host=${spring.data.redis.host} +sa-token.alone-redis.port=${spring.data.redis.port} +sa-token.alone-redis.password=${spring.data.redis.password} +sa-token.alone-redis.timeout=${spring.data.redis.timeout} +sa-token.alone-redis.lettuce.pool.max-active=${spring.data.redis.lettuce.pool.max-active} +sa-token.alone-redis.lettuce.pool.max-wait=${spring.data.redis.lettuce.pool.max-wait} +sa-token.alone-redis.lettuce.pool.max-idle=${spring.data.redis.lettuce.pool.max-idle} +sa-token.alone-redis.lettuce.pool.min-idle=${spring.data.redis.lettuce.pool.min-idle} + +######################################### +# knife4j configuration +######################################### +knife4j.enable=true +knife4j.production=false +knife4j.basic.enable=true +knife4j.basic.username=admin +knife4j.basic.password=123456 +knife4j.setting.enableOpenApi=false +knife4j.setting.enableSwaggerModels=false +knife4j.setting.enableFooter=false +knife4j.setting.enableFooterCustom=true +knife4j.setting.footerCustomContent=Apache License 2.0 | Copyright 2020-2024[SNOWY](https://www.xiaonuo.vip) +springdoc.default-flat-param-object=true + +# knife4j doc groups +springdoc.group-configs[0].group=SNOWY-PLUGIN-AUTH +springdoc.group-configs[0].display-name=${springdoc.group-configs[0].group} +springdoc.group-configs[0].packages-to-scan=vip.xiaonuo.auth +springdoc.group-configs[1].group=SNOWY-PLUGIN-BIZ +springdoc.group-configs[1].display-name=${springdoc.group-configs[1].group} +springdoc.group-configs[1].packages-to-scan=vip.xiaonuo.biz +springdoc.group-configs[2].group=SNOWY-PLUGIN-CLIENT +springdoc.group-configs[2].display-name=${springdoc.group-configs[2].group} +springdoc.group-configs[2].packages-to-scan=vip.xiaonuo.client +springdoc.group-configs[3].group=SNOWY-PLUGIN-DEV +springdoc.group-configs[3].display-name=${springdoc.group-configs[3].group} +springdoc.group-configs[3].packages-to-scan=vip.xiaonuo.dev +springdoc.group-configs[4].group=SNOWY-PLUGIN-GEN +springdoc.group-configs[4].display-name=${springdoc.group-configs[4].group} +springdoc.group-configs[4].packages-to-scan=vip.xiaonuo.gen +springdoc.group-configs[5].group=SNOWY-PLUGIN-MOBILE +springdoc.group-configs[5].display-name=${springdoc.group-configs[5].group} +springdoc.group-configs[5].packages-to-scan=vip.xiaonuo.mobile +springdoc.group-configs[6].group=SNOWY-PLUGIN-SYS +springdoc.group-configs[6].display-name=${springdoc.group-configs[6].group} +springdoc.group-configs[6].packages-to-scan=vip.xiaonuo.sys + +######################################### +# snowy configuration +######################################### + +# common configuration +snowy.config.common.backend-url=http://localhost:82 + +# plugin dev-sms configuration +sms-oa.config-type=yaml +sms-oa.core-pool-size=20 +sms-oa.queue-capacity=20 +sms-oa.max-pool-size=20 + diff --git a/snowy-web-app/src/main/resources/application.yaml b/snowy-web-app/src/main/resources/application.yaml deleted file mode 100644 index b0e464bf..00000000 --- a/snowy-web-app/src/main/resources/application.yaml +++ /dev/null @@ -1,163 +0,0 @@ -server: - port: 82 # 服务端口 - -spring: - profiles: - active: local # 本地环境 - # active: test # 测试环境 - # active: prod # 生产环境 - servlet: - multipart: - max-request-size: 100MB # 定义单个HTTP请求的最大体积 - max-file-size: 100MB # 定义单个上传文件的最大体积限制 - main: - allow-circular-references: true # 允许循环引用 - # jackson 配置 - jackson: - time-zone: GMT+8 - date-format: yyyy-MM-dd HH:mm:ss - locale: zh_CN - serialization: - write-dates-as-timestamps: false - # 数据源相关配置 - datasource: - dynamic: - datasource: - master: # 主数据源配置 - driver-class-name: com.mysql.cj.jdbc.Driver - url: jdbc:mysql://localhost:3306/snowy?useUnicode=true&characterEncoding=utf-8&useSSL=false&allowPublicKeyRetrieval=true&nullCatalogMeansCurrent=true&useInformationSchema=true&rewriteBatchedStatements=true - username: root - password: 123456 - strict: true - public-key: MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAMWiTVtdXFVrgFHDDKELZM0SywkWY3KjugN90eY5Sogon1j8Y0ClPF7nx3FuE7pAeBKiv7ChIS0vvx/59WUpKmUCAwEAAQ== - # Druid连接池参数(针对动态数据源) - druid: - initial-size: 5 - max-active: 20 - min-idle: 5 - max-wait: 60000 - pool-prepared-statements: true - max-pool-prepared-statement-per-connection-size: 20 - validation-query-timeout: 2000 - test-on-borrow: false - test-on-return: false - test-while-idle: true - time-between-eviction-runs-millis: 60000 - min-evictable-idle-time-millis: 300000 - filters: stat - break-after-acquire-failure: false - # Druid全局配置(监控界面) - druid: - stat-view-servlet: - enabled: true - login-username: admin - login-password: 123456 - - # redis 配置 - data: - redis: - database: 1 - host: 127.0.0.1 - port: 6379 - password: - timeout: 10s - lettuce: - pool: - max-active: 200 - max-wait: -1ms - max-idle: 10 - min-idle: 0 - -# mybatis-plus 配置 -mybatis-plus: - configuration: - log-impl: org.apache.ibatis.logging.stdout.StdOutImpl - jdbc-type-for-null: null - global-config: - banner: false - enable-sql-runner: true - db-config: - id-type: ASSIGN_ID - logic-delete-field: DELETE_FLAG - logic-delete-value: DELETED - logic-not-delete-value: NOT_DELETE - mapper-locations: classpath*:vip/xiaonuo/**/mapping/*.xml - type-handlers-package: vip.xiaonuo.common.handler - -# easy-trans 配置 -easy-trans: - is-enable-redis: true - is-enable-global: true - is-enable-tile: true - is-enable-cloud: false - -# sa-token 配置 -sa-token: - token-name: token - timeout: 2592000 - active-timeout: -1 - is-concurrent: true - is-share: false - max-login-count: -1 - token-style: random-32 - is-log: false - is-print: false - alone-redis: - database: 2 - host: ${spring.data.redis.host} - port: ${spring.data.redis.port} - password: ${spring.data.redis.password} - timeout: ${spring.data.redis.timeout} - lettuce: - pool: - max-active: ${spring.data.redis.lettuce.pool.max-active} - max-wait: ${spring.data.redis.lettuce.pool.max-wait} - max-idle: ${spring.data.redis.lettuce.pool.max-idle} - min-idle: ${spring.data.redis.lettuce.pool.min-idle} - -# knife4j 配置 -knife4j: - enable: true - production: false - basic: - enable: true - username: admin - password: 123456 - setting: - enableOpenApi: false - enableSwaggerModels: false - enableFooter: false - enableFooterCustom: true - footerCustomContent: Apache License 2.0 | Copyright 2020-2024[SNOWY](https://www.xiaonuo.vip) - -springdoc: - default-flat-param-object: true - group-configs: - - group: SNOWY-PLUGIN-AUTH - display-name: SNOWY-PLUGIN-AUTH - packages-to-scan: vip.xiaonuo.auth - - group: SNOWY-PLUGIN-BIZ - display-name: SNOWY-PLUGIN-BIZ - packages-to-scan: vip.xiaonuo.biz - - group: SNOWY-PLUGIN-CLIENT - display-name: SNOWY-PLUGIN-CLIENT - packages-to-scan: vip.xiaonuo.client - - group: SNOWY-PLUGIN-DEV - display-name: SNOWY-PLUGIN-DEV - packages-to-scan: vip.xiaonuo.dev - - group: SNOWY-PLUGIN-GEN - display-name: SNOWY-PLUGIN-GEN - packages-to-scan: vip.xiaonuo.gen - - group: SNOWY-PLUGIN-MOBILE - display-name: SNOWY-PLUGIN-MOBILE - packages-to-scan: vip.xiaonuo.mobile - - group: SNOWY-PLUGIN-SYS - display-name: SNOWY-PLUGIN-SYS - packages-to-scan: vip.xiaonuo.sys - -# snowy 配置 -snowy: - config: - common: - front-url: http://localhost:${server.port} - backend-url: http://localhost:${server.port} \ No newline at end of file diff --git a/snowy-web-app/src/main/resources/favicon.ico b/snowy-web-app/src/main/resources/favicon.ico new file mode 100644 index 00000000..9e3ed92f Binary files /dev/null and b/snowy-web-app/src/main/resources/favicon.ico differ