|
@@ -1,47 +1,48 @@
|
|
|
package com.stellvoy.user.service.impl;
|
|
|
|
|
|
import cn.hutool.core.util.StrUtil;
|
|
|
+import com.alibaba.fastjson.JSON;
|
|
|
+import com.alibaba.fastjson.JSONObject;
|
|
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
|
|
import com.stellvoy.api.client.auth.AuthClient;
|
|
|
import com.stellvoy.api.client.task.TaskClient;
|
|
|
-import com.stellvoy.api.dto.auth.RoleDTO;
|
|
|
+import com.stellvoy.api.client.user.UserClient;
|
|
|
import com.stellvoy.api.dto.task.NewTaskDTO;
|
|
|
import com.stellvoy.api.dto.user.LoginFormDTO;
|
|
|
-import com.stellvoy.api.dto.user.UserDTO;
|
|
|
import com.stellvoy.common.domain.dto.LoginUserDTO;
|
|
|
-import com.stellvoy.common.enums.UserType;
|
|
|
import com.stellvoy.common.exceptions.BadRequestException;
|
|
|
import com.stellvoy.common.exceptions.ForbiddenException;
|
|
|
-import com.stellvoy.common.exceptions.UnauthorizedException;
|
|
|
import com.stellvoy.common.utils.AssertUtils;
|
|
|
import com.stellvoy.common.utils.BeanUtils;
|
|
|
import com.stellvoy.common.utils.StringUtils;
|
|
|
import com.stellvoy.common.utils.UserContext;
|
|
|
import com.stellvoy.user.domain.dto.*;
|
|
|
import com.stellvoy.user.domain.po.User;
|
|
|
-import com.stellvoy.user.domain.po.UserDetail;
|
|
|
import com.stellvoy.user.domain.po.UserTask;
|
|
|
-import com.stellvoy.user.domain.vo.UserDetailVO;
|
|
|
import com.stellvoy.user.mapper.UserMapper;
|
|
|
import com.stellvoy.user.service.ICodeService;
|
|
|
-import com.stellvoy.user.service.IUserDetailService;
|
|
|
import com.stellvoy.user.service.IUserService;
|
|
|
import com.stellvoy.user.service.UserTaskService;
|
|
|
import com.stellvoy.user.utils.AliyunIdCardVerifier;
|
|
|
+import lombok.extern.slf4j.Slf4j;
|
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
|
+import org.springframework.beans.factory.annotation.Value;
|
|
|
import org.springframework.data.redis.core.StringRedisTemplate;
|
|
|
-import org.springframework.security.crypto.password.PasswordEncoder;
|
|
|
+import org.springframework.http.ResponseEntity;
|
|
|
+import org.springframework.http.converter.StringHttpMessageConverter;
|
|
|
import org.springframework.stereotype.Service;
|
|
|
-import org.springframework.transaction.annotation.Transactional;
|
|
|
+import org.springframework.web.client.RestTemplate;
|
|
|
|
|
|
+
|
|
|
+import java.io.IOException;
|
|
|
+import java.io.UnsupportedEncodingException;
|
|
|
+import java.net.URLEncoder;
|
|
|
+import java.nio.charset.StandardCharsets;
|
|
|
import java.time.LocalDateTime;
|
|
|
-import java.time.Month;
|
|
|
-import java.util.HashMap;
|
|
|
-import java.util.List;
|
|
|
-import java.util.Map;
|
|
|
+import java.util.*;
|
|
|
+import java.util.concurrent.TimeUnit;
|
|
|
import java.util.stream.Collectors;
|
|
|
|
|
|
-import static com.stellvoy.user.constants.UserConstants.*;
|
|
|
import static com.stellvoy.user.constants.UserErrorInfo.Msg.*;
|
|
|
|
|
|
|
|
@@ -51,17 +52,12 @@ import static com.stellvoy.user.constants.UserErrorInfo.Msg.*;
|
|
|
* </p>
|
|
|
*/
|
|
|
@Service
|
|
|
+@Slf4j
|
|
|
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IUserService {
|
|
|
|
|
|
@Autowired
|
|
|
- private PasswordEncoder passwordEncoder;
|
|
|
- @Autowired
|
|
|
private ICodeService codeService;
|
|
|
@Autowired
|
|
|
- private AuthClient authClient;
|
|
|
- @Autowired
|
|
|
- private IUserDetailService detailService;
|
|
|
- @Autowired
|
|
|
private AliyunIdCardVerifier aliyunIdCardVerifier;
|
|
|
@Autowired
|
|
|
private StringRedisTemplate stringRedisTemplate;
|
|
@@ -69,6 +65,15 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
private UserTaskService userTaskService;
|
|
|
@Autowired
|
|
|
private TaskClient taskClient;
|
|
|
+ @Autowired
|
|
|
+ private AuthClient authClient;
|
|
|
+ @Value("${wechat.appid}")
|
|
|
+ private String appId;
|
|
|
+
|
|
|
+ @Value("${wechat.appsecret}")
|
|
|
+ private String appSecret;
|
|
|
+ private final RestTemplate restTemplate = new RestTemplate();
|
|
|
+
|
|
|
@Override
|
|
|
public LoginUserDTO queryUserDetail(LoginFormDTO loginDTO, boolean isStaff) {
|
|
|
// 1.判断登录方式
|
|
@@ -82,6 +87,11 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
if (type == 2) {
|
|
|
user = loginByVerifyCode(loginDTO.getCellPhone(), loginDTO.getPassword());
|
|
|
}
|
|
|
+ // 微信登录
|
|
|
+ if (type == 3) {
|
|
|
+ int userId = Integer.parseInt(loginDTO.getUsername());
|
|
|
+ user = lambdaQuery().eq(User::getId, userId).one();
|
|
|
+ }
|
|
|
// 4.错误的登录方式
|
|
|
if (user == null) {
|
|
|
throw new BadRequestException(ILLEGAL_LOGIN_TYPE);
|
|
@@ -93,157 +103,6 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
return userDTO;
|
|
|
}
|
|
|
|
|
|
- @Override
|
|
|
- public void resetPassword(Long userId) {
|
|
|
- User user = new User();
|
|
|
- user.setId(userId);
|
|
|
- user.setPassword(passwordEncoder.encode(DEFAULT_PASSWORD));
|
|
|
- AssertUtils.isTrue(updateById(user), USER_ID_NOT_EXISTS);
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- public UserDetailVO myInfo() {
|
|
|
- // 1.获取登录用户id
|
|
|
- Long userId = UserContext.getUser();
|
|
|
- if (userId == null) {
|
|
|
- return null;
|
|
|
- }
|
|
|
- // 2.查询用户
|
|
|
- UserDetail userDetail = detailService.queryById(userId);
|
|
|
- AssertUtils.isNotNull(userDetail, USER_ID_NOT_EXISTS);
|
|
|
- // 3.封装vo
|
|
|
- UserType type = userDetail.getType();
|
|
|
- // 3.1.基本信息
|
|
|
- UserDetailVO vo = BeanUtils.toBean(userDetail, UserDetailVO.class);
|
|
|
- // 3.2.详情信息
|
|
|
- switch (type) {
|
|
|
- case STAFF:
|
|
|
- RoleDTO roleDTO = authClient.queryRoleById(userDetail.getRoleId());
|
|
|
- vo.setRoleName(roleDTO == null ? "" : roleDTO.getName());
|
|
|
- break;
|
|
|
- case STUDENT:
|
|
|
- vo.setRoleName(STUDENT_ROLE_NAME);
|
|
|
- break;
|
|
|
- case TEACHER:
|
|
|
- vo.setRoleName(TEACHER_ROLE_NAME);
|
|
|
- break;
|
|
|
- default:
|
|
|
- break;
|
|
|
- }
|
|
|
- return vo;
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- public void addUserByPhone(User user, String code) {
|
|
|
- // 1.验证码校验
|
|
|
- codeService.verifyCode(user.getCellPhone(), code);
|
|
|
- // 2.判断手机号是否存在
|
|
|
- Long count = lambdaQuery().eq(User::getCellPhone, user.getCellPhone()).count();
|
|
|
- if (count > 0) {
|
|
|
- throw new BadRequestException(PHONE_ALREADY_EXISTS);
|
|
|
- }
|
|
|
- // 3.加密密码
|
|
|
- user.setPassword(passwordEncoder.encode(user.getPassword()));
|
|
|
- // 4.新增
|
|
|
- user.setUsername(user.getCellPhone());
|
|
|
- save(user);
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- public void updatePasswordByPhone(String cellPhone, String code, String password) {
|
|
|
- // 1.验证码校验
|
|
|
- codeService.verifyCode(cellPhone, code);
|
|
|
- // 2.查询用户
|
|
|
- User oldUser = lambdaQuery().eq(User::getCellPhone, cellPhone).one();
|
|
|
- if (oldUser == null) {
|
|
|
- // 手机号不存在
|
|
|
- throw new BadRequestException(PHONE_NOT_EXISTS);
|
|
|
- }
|
|
|
- // 2.修改密码
|
|
|
- User user = new User();
|
|
|
- user.setId(user.getId());
|
|
|
- user.setPassword(passwordEncoder.encode(password));
|
|
|
- updateById(user);
|
|
|
- }
|
|
|
-
|
|
|
- public void updatePhoneById(Long id, String cellPhone) {
|
|
|
- // 1.1.判断是否需要修改手机号
|
|
|
- if (StringUtils.isNotBlank(cellPhone)) {
|
|
|
- // 1.2.需要修改,封装数据
|
|
|
- User user = new User();
|
|
|
- user.setId(id);
|
|
|
- user.setUsername(cellPhone);
|
|
|
- user.setCellPhone(cellPhone);
|
|
|
- // 1.3.修改
|
|
|
- updateById(user);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- @Transactional
|
|
|
- public Long saveUser(UserDTO userDTO) {
|
|
|
- // 1.保存用户基本信息
|
|
|
- User user = new User();
|
|
|
- user.setPassword(passwordEncoder.encode(DEFAULT_PASSWORD));
|
|
|
- user.setCellPhone(userDTO.getCellPhone());
|
|
|
- user.setUsername(userDTO.getCellPhone());
|
|
|
- save(user);
|
|
|
- // 2.新增详情
|
|
|
- UserDetail detail = BeanUtils.toBean(userDTO, UserDetail.class);
|
|
|
- detail.setId(user.getId());
|
|
|
- detailService.save(detail);
|
|
|
- return user.getId();
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- @Transactional
|
|
|
- public void updateUser(UserDTO userDTO) {
|
|
|
- // 1.如果传递了手机号,则修改手机号
|
|
|
- String cellphone = userDTO.getCellPhone();
|
|
|
- if(StringUtils.isNotBlank(cellphone)){
|
|
|
- User user = new User();
|
|
|
- user.setId(userDTO.getId());
|
|
|
- user.setCellPhone(cellphone);
|
|
|
- user.setUsername(cellphone);
|
|
|
- updateById(user);
|
|
|
- }
|
|
|
- // 2.修改详情
|
|
|
- UserDetail detail = BeanUtils.toBean(userDTO, UserDetail.class);
|
|
|
- detail.setType(null);
|
|
|
- detailService.updateById(detail);
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- public void updateUserWithPassword(UserFormDTO userDTO) {
|
|
|
- // 1.尝试更新密码
|
|
|
- String pw = userDTO.getPassword();
|
|
|
- String oldPw = userDTO.getOldPassword();
|
|
|
- if(StringUtils.isNotBlank(pw) && StringUtils.isNotBlank(pw)) {
|
|
|
- Long userId = UserContext.getUser();
|
|
|
- // 1.1.查询用户
|
|
|
- User user = getById(userId);
|
|
|
- // 1.2.校验
|
|
|
- if (user == null) {
|
|
|
- throw new UnauthorizedException(USER_ID_NOT_EXISTS);
|
|
|
- }
|
|
|
- // 1.3.校验密码
|
|
|
- if (!passwordEncoder.matches(oldPw, user.getPassword())) {
|
|
|
- // 密码不一致
|
|
|
- throw new UnauthorizedException(INVALID_UN_OR_PW);
|
|
|
- }
|
|
|
- // 1.4.修改密码
|
|
|
- user = new User();
|
|
|
- user.setId(userId);
|
|
|
- user.setPassword(passwordEncoder.encode(pw));
|
|
|
- updateById(user);
|
|
|
- }
|
|
|
- // 2.更新用户详情
|
|
|
- UserDetail detail = BeanUtils.toBean(userDTO, UserDetail.class);
|
|
|
- detail.setRoleId(null);
|
|
|
- detail.setType(null);
|
|
|
- detailService.updateById(detail);
|
|
|
- }
|
|
|
-
|
|
|
/**
|
|
|
* 手机号注册
|
|
|
*/
|
|
@@ -263,6 +122,9 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
user.setUpdateTime(LocalDateTime.now());
|
|
|
user.setCreater(UserContext.getUser());
|
|
|
user.setUpdater(UserContext.getUser());
|
|
|
+ user.setUsername(dto.getCellPhone());
|
|
|
+ user.setNickName(dto.getCellPhone());
|
|
|
+ user.setImage("https://suppliermanage.oss-cn-shanghai.aliyuncs.com/gig/img/woman.png");
|
|
|
save(user);
|
|
|
return user;
|
|
|
}
|
|
@@ -279,12 +141,24 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
} else {
|
|
|
throw new RuntimeException("身份证号码不能为空");
|
|
|
}
|
|
|
+ int sex = 0;
|
|
|
+ String image = "https://suppliermanage.oss-cn-shanghai.aliyuncs.com/gig/img/woman.png";
|
|
|
+ if(!sex(dto.getIdentityCardId())){
|
|
|
+ sex = 1;
|
|
|
+ image = "https://suppliermanage.oss-cn-shanghai.aliyuncs.com/gig/img/man.png";
|
|
|
+ }
|
|
|
lambdaUpdate()
|
|
|
.eq(User::getId, userId)
|
|
|
+ .set(User::getNickName, dto.getName())
|
|
|
.set(User::getIdentityCardId, dto.getIdentityCardId())
|
|
|
+ .set(User::getSex, sex)
|
|
|
+ .set(User::getImage, image)
|
|
|
.update();
|
|
|
}
|
|
|
-
|
|
|
+ private boolean sex(String getIdentityCardId) {
|
|
|
+ char[] charArray = getIdentityCardId.toCharArray();
|
|
|
+ return charArray[16] % 2 == 0;
|
|
|
+ }
|
|
|
/**
|
|
|
* 修改登录密码
|
|
|
*/
|
|
@@ -325,6 +199,12 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
*/
|
|
|
public UserTaskDTO getStats(Long userId) {
|
|
|
List<UserTask> userTaskList = userTaskService.lambdaQuery().eq(UserTask::getUserId, userId).list();
|
|
|
+
|
|
|
+ // 检查 userTaskList 是否为空
|
|
|
+ if (userTaskList == null || userTaskList.isEmpty()) {
|
|
|
+ return new UserTaskDTO();
|
|
|
+ }
|
|
|
+
|
|
|
UserTaskDTO userTaskDTO = new UserTaskDTO();
|
|
|
userTaskDTO.setOngoingTasks(0);
|
|
|
userTaskDTO.setSettledTasks(0);
|
|
@@ -358,13 +238,126 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
.filter(userTask -> userTask.getTaskStatus() == 3)
|
|
|
.map(UserTask::getTaskId)
|
|
|
.collect(Collectors.toList());
|
|
|
+
|
|
|
+ // 检查 ids 是否为空
|
|
|
+ if (ids.isEmpty()) {
|
|
|
+ userTaskDTO.setTaskDetails(new ArrayList<>());
|
|
|
+ return userTaskDTO;
|
|
|
+ }
|
|
|
+
|
|
|
List<NewTaskDTO> newTaskDTOS = taskClient.queryTaskById(ids);
|
|
|
- List<TaskDetail> taskDetails = BeanUtils.copyList(newTaskDTOS, TaskDetail.class);
|
|
|
- userTaskDTO.setTaskDetails(taskDetails);
|
|
|
+
|
|
|
+ // 检查 newTaskDTOS 是否为空
|
|
|
+ if (newTaskDTOS == null || newTaskDTOS.isEmpty()) {
|
|
|
+ userTaskDTO.setTaskDetails(new ArrayList<>());
|
|
|
+ } else {
|
|
|
+ List<TaskDetail> taskDetails = BeanUtils.copyList(newTaskDTOS, TaskDetail.class);
|
|
|
+ userTaskDTO.setTaskDetails(taskDetails);
|
|
|
+ }
|
|
|
+
|
|
|
return userTaskDTO;
|
|
|
}
|
|
|
|
|
|
+ /**
|
|
|
+ * 生成微信登录二维码
|
|
|
+ */
|
|
|
+ public WeChatDTO wechatLogin() throws UnsupportedEncodingException {
|
|
|
+ WeChatDTO weChatDTO = new WeChatDTO();
|
|
|
+ String redirectUri = "https://labelx.stellvoy.com/us/users/wechat/code";
|
|
|
+ String encodedRedirectUri = URLEncoder.encode(redirectUri, StandardCharsets.UTF_8.toString());
|
|
|
+ String state = UUID.randomUUID().toString();
|
|
|
+ weChatDTO.setRedirectUri(encodedRedirectUri);
|
|
|
+ weChatDTO.setState(state);
|
|
|
+ weChatDTO.setAppid(appId);
|
|
|
+ stringRedisTemplate.opsForValue().set("weChatCode:" + state, state, 5, TimeUnit.MINUTES);
|
|
|
+ return weChatDTO;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 处理微信回调,接收 code
|
|
|
+ */
|
|
|
+ public void wechatCallback(String code, String state) {
|
|
|
+ // 从Redis中获取state
|
|
|
+ String storedCode = stringRedisTemplate.opsForValue().get("weChatCode:" + state);
|
|
|
+ // state校验
|
|
|
+ if (storedCode == null || !storedCode.equals(state)) {
|
|
|
+ throw new RuntimeException("state错误或已过期");
|
|
|
+ }
|
|
|
+ // 使用 code 获取 access_token 和 openid
|
|
|
+ String accessTokenUrl = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=" + appId +
|
|
|
+ "&secret=" + appSecret +
|
|
|
+ "&code=" + code +
|
|
|
+ "&grant_type=authorization_code";
|
|
|
+ RestTemplate restTemplate = new RestTemplate();
|
|
|
+ restTemplate.getMessageConverters().add(0, new StringHttpMessageConverter(StandardCharsets.UTF_8));
|
|
|
+ ResponseEntity<String> accessTokenResponse = restTemplate.getForEntity(accessTokenUrl, String.class);
|
|
|
+ JSONObject accessTokenJson = JSON.parseObject(accessTokenResponse.getBody());
|
|
|
+ String accessToken = accessTokenJson.getString("access_token");
|
|
|
+ String openid = accessTokenJson.getString("openid");
|
|
|
+ // 使用 access_token 和 openid 获取用户信息
|
|
|
+ String userInfoUrl = "https://api.weixin.qq.com/sns/userinfo?access_token=" + accessToken +
|
|
|
+ "&openid=" + openid +
|
|
|
+ "&lang=zh_CN";
|
|
|
+ ResponseEntity<String> userInfoResponse = restTemplate.getForEntity(userInfoUrl, String.class);
|
|
|
+ JSONObject userInfoJson = JSON.parseObject(userInfoResponse.getBody());
|
|
|
+
|
|
|
+ // 处理用户信息,例如登录或注册用户
|
|
|
+ String nickname = userInfoJson.getString("nickname");
|
|
|
+ String headimgurl = userInfoJson.getString("headimgurl");
|
|
|
+ User user = lambdaQuery().eq(User::getOpenId, openid).one();
|
|
|
+ if(user == null){
|
|
|
+ user = new User();
|
|
|
+ user.setNickName(nickname);
|
|
|
+ user.setImage(headimgurl);
|
|
|
+ user.setOpenId(openid);
|
|
|
+ user.setPassword("123456");
|
|
|
+ save(user);
|
|
|
+ user = lambdaQuery().eq(User::getOpenId, openid).one();
|
|
|
+ }
|
|
|
+ stringRedisTemplate.opsForValue().set("weChatCode:" + state, openid, 5, TimeUnit.MINUTES);
|
|
|
+ stringRedisTemplate.opsForValue().set("userId:" + state, user.getId().toString(), 5, TimeUnit.MINUTES);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 处理前端轮询
|
|
|
+ */
|
|
|
+ public CheckLoginDTO checkLoginStatus(String state) {
|
|
|
+ // 从Redis中获取state
|
|
|
+ String storedCode = stringRedisTemplate.opsForValue().get("weChatCode:" + state);
|
|
|
+ // state校验
|
|
|
+ if (storedCode == null) {
|
|
|
+ throw new RuntimeException("state错误或已失效");
|
|
|
+ }
|
|
|
+ User one = lambdaQuery().eq(User::getOpenId, storedCode).one();
|
|
|
+ if (one == null) {
|
|
|
+ throw new RuntimeException("用户未登录");
|
|
|
+ }
|
|
|
+ LoginFormDTO loginDTO = new LoginFormDTO();
|
|
|
+ loginDTO.setType(3);
|
|
|
+ loginDTO.setPassword("123456");
|
|
|
+ String userId = stringRedisTemplate.opsForValue().get("userId:" + state);
|
|
|
+ loginDTO.setUsername(userId);
|
|
|
+ String token = authClient.loginByPw(loginDTO);
|
|
|
+ CheckLoginDTO checkLoginDTO = new CheckLoginDTO();
|
|
|
+ checkLoginDTO.setToken(token);
|
|
|
+ checkLoginDTO.setUserId(one.getId());
|
|
|
+ if(one.getCellPhone()==null){
|
|
|
+ checkLoginDTO.setCellPhone("未绑定手机号");
|
|
|
+ }
|
|
|
+ checkLoginDTO.setCellPhone(one.getCellPhone());
|
|
|
+ return checkLoginDTO;
|
|
|
+ }
|
|
|
|
|
|
+ /**
|
|
|
+ * 验证验证码
|
|
|
+ */
|
|
|
+ public void weChatVerifyCode(WeChatVerifyCodeDTO dto, Long userId) {
|
|
|
+ codeService.verifyCode(dto.getPhone(), dto.getCode());
|
|
|
+ lambdaUpdate().eq(User::getId, userId)
|
|
|
+ .set(User::getCellPhone, dto.getPhone())
|
|
|
+ .set(User::getUsername, dto.getPhone())
|
|
|
+ .update();
|
|
|
+ }
|
|
|
|
|
|
public User loginByPw(LoginFormDTO loginDTO) {
|
|
|
// 1.数据校验
|
|
@@ -381,7 +374,7 @@ public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IU
|
|
|
if (user.getStatus() == 0) {
|
|
|
throw new ForbiddenException(USER_FROZEN);
|
|
|
}
|
|
|
- // 4.校验密码
|
|
|
+ //4.校验密码
|
|
|
// if (!passwordEncoder.matches(loginDTO.getPassword(), user.getPassword())) {
|
|
|
// throw new BadRequestException(INVALID_UN_OR_PW);
|
|
|
// }
|