1、注册模块/register
get请求时,直接返回 "/site/register"界面。post请求提交时,调用service层的register方法,进行表单校验(空参,账号密码、邮箱)
空参直接返回Exception,其他错误信息封装在map中,返回map给controller和前端。
注册用户:封装用户信息,调用userMapper.insertUser(user)添加用户,状态设置未激活,随机生成一个字符串加到密码尾部,使用MD5加密技术对密码进行加密。
激活邮件:
为防止重复登录以及伪造激活链接,在user内封装一个随机的UUID作为激活码,将激活码附在URL上,点击激活链接进行激活,比较激活码是否相同,并修改数据库中的用户状态为0->1,涉及到数据的更改,先更新数据库,再删除Redis缓存。激活成功,则跳转回登录页面,失败跳转回首页。
---------------------------------------------------------------------------------------------------------------------------------
1.UserMapper:
package com.nowcoder.community.mapper;
import com.nowcoder.community.entity.User;
import org.apache.ibatis.annotations.Param;
import org.springframework.stereotype.Repository;
@Repository
public interface UserMapper {
User selectById(int id);
User selectByName(String username);
User selectByEmail(String email);
int insertUser(User user);
int updateStatus(@Param("id") int id, @Param("status") int status);
int updateHeader(int id,String headerUrl);
int updatePassword(int id,String password);
}
2.编写UserMapper.xml:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.nowcoder.community.mapper.UserMapper">
<select id="selectById" resultType="user" parameterType="_int">
select * from user where id=#{id};
</select>
<select id="selectByName" resultType="user">
select * from user where username=#{username};
</select>
<select id="selectByEmail" resultType="user">
select * from user where email=#{email};
</select>
<insert id="insertUser" parameterType="user" keyProperty="id">
insert into user(username, password, salt, email, type, status, activation_code, header_url, create_time) VALUES (#{username},#{password},#{salt},#{email},#{type},#{status},#{activationCode},#{headerUrl},#{createTime});
</insert>
<update id="updateStatus">
update user set status=#{status} where id=#{id}
</update>
<update id="updateHeader">
update user set header_url=#{headerUrl} where id=#{id};
</update>
<update id="updatePassword">
update user set password=#{password} where id=#{id};
</update>
</mapper>
3.UserService:
package com.nowcoder.community.service;
import com.nowcoder.community.entity.LoginTicket;
import com.nowcoder.community.entity.User;
import com.nowcoder.community.mapper.LoginTickerMapper;
import com.nowcoder.community.mapper.UserMapper;
import com.nowcoder.community.util.CommunityConstant;
import com.nowcoder.community.util.CommunityUtil;
import com.nowcoder.community.util.MailClient;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
import org.thymeleaf.TemplateEngine;
import org.thymeleaf.context.Context;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.Random;
@Service
public class UserServiceImpl implements UserService, CommunityConstant {
@Autowired
private UserMapper userMapper;
@Autowired
private MailClient mailClient;
@Autowired
private TemplateEngine templateEngine;
@Value("${community.path.domain}")
private String domain;
@Value("${server.servlet.context-path}")
private String contextPath;
public User selectById(int id) {
return userMapper.selectById(id);
}
@Override
public User selectByName(String username) {
return userMapper.selectByName(username);
}
@Override
public User selectByEmail(String email) {
return userMapper.selectByEmail(email);
}
@Override
public int insertUser(User user) {
return userMapper.insertUser(user);
}
@Override
public int updateStatus(int id, int status) {
return userMapper.updateStatus(id, status);
}
@Override
public int updateHeader(int id, String headerUrl) {
return userMapper.updateHeader(id, headerUrl);
}
@Override
public int updatePassword(int id, String password) {
return userMapper.updatePassword(id, password);
}
/**
* 注册账号
* @param user
* @return
*/
public Map<String, Object> register(User user) {
Map<String, Object> map = new HashMap<>();
// 空值处理
if (user == null) {
throw new IllegalArgumentException("参数不能为空!");
}
if (StringUtils.isBlank(user.getUsername())) {
map.put("usernameMsg", "账号不能为空!");
return map;
}
if (StringUtils.isBlank(user.getPassword())) {
map.put("passwordMsg", "密码不能为空!");
return map;
}
if (StringUtils.isBlank(user.getEmail())) {
map.put("emailMsg", "邮箱不能为空!");
return map;
}
// 验证账号
User u = userMapper.selectByName(user.getUsername());
if (u != null) {
map.put("usernameMsg", "该账号已存在!");
return map;
}
// 验证邮箱
u = userMapper.selectByEmail(user.getEmail());
if (u != null) {
map.put("emailMsg", "该邮箱已被注册!");
return map;
}
// 注册用户
user.setSalt(CommunityUtil.generateUUID().substring(0, 5));
user.setPassword(CommunityUtil.md5(user.getPassword() + user.getSalt()));
user.setType(0);
user.setStatus(0);
user.setActivationCode(CommunityUtil.generateUUID());
user.setHeaderUrl(String.format("http://images.nowcoder.com/head/%dt.png", new Random().nextInt(1000)));
user.setCreateTime(new Date());
userMapper.insertUser(user);
// 激活邮件
Context context = new Context();
context.setVariable("email", user.getEmail());
// http://localhost:8080/community/activation/101/code
String url = domain + contextPath + "/activation/" + user.getId() + "/" + user.getActivationCode();
context.setVariable("url", url);
String content = templateEngine.process("/mail/activation", context);
mailClient.sendMail(user.getEmail(), "激活账号", content);
return map;
}
/**
* 激活账号
* @param userId
* @param code
* @return
*/
public int activation(int userId,String code){
User user = userMapper.selectById(userId);
if (user.getStatus()==1){
return ACTIVATION_REPEAT;
}
else if (user.getActivationCode().equals(code)){
userMapper.updateStatus(userId,1);
return ACTIVATION_SUCCESS;
}
else{
return ACTIVATION_FAILURE;
}
}
4.LoginController:
package com.nowcoder.community.controller;
import com.google.code.kaptcha.Producer;
import com.nowcoder.community.entity.User;
import com.nowcoder.community.service.UserServiceImpl;
import com.nowcoder.community.util.CommunityConstant;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.CookieValue;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import javax.imageio.ImageIO;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Map;
@Controller
public class LoginController implements CommunityConstant {
private static final Logger logger= LoggerFactory.getLogger(LoginController.class);
@Autowired
private UserServiceImpl userService;
@Value("${server.servlet.context-path}")
private String contextPath;
@GetMapping("/register")
public String getRegisterPage(){
return "/site/register";
}
@PostMapping("/register")
public String register(Model model, User user){
Map<String,Object> map=userService.register(user);
if (map==null|| map.isEmpty()){
model.addAttribute("msg","注册成功,我们已经向您的邮箱发送了一封激活邮件,请尽快激活!");
model.addAttribute("target","/index");
return "/site/operate-result";
}else {
model.addAttribute("usernameMsg",map.get("usernameMsg"));
model.addAttribute("passwordMsg",map.get("passwordMsg"));
model.addAttribute("emailMsg",map.get("emailMsg"));
return "/site/register";
}
}
// http://localhost:8080/community/activation/101/code
@GetMapping("/activation/{userId}/{code}")
public String activation(Model model, @PathVariable("userId") int userId, @PathVariable("code") String code){
int result = userService.activation(userId, code);
if (result==ACTIVATION_SUCCESS){
model.addAttribute("msg","激活成功,您的帐号已经可以正常使用!");
model.addAttribute("target","/login");
}
else if (result==ACTIVATION_REPEAT){
model.addAttribute("msg","无效操作,该账号已经激活过!");
model.addAttribute("target","/index");
}
else {
model.addAttribute("msg","激活失败,您提供的激活码不正确!");
model.addAttribute("target","/index");
}
return "/site/operate-result";
5.编写常量工具类CommunityConstant:
package com.nowcoder.community.util;
public interface CommunityConstant {
/**
* 激活成功
*/
int ACTIVATION_SUCCESS=0;
/**
*重复激活
*/
int ACTIVATION_REPEAT=1;
/**
* 激活失败
*/
int ACTIVATION_FAILURE=2;
/**
* 默认状态的登陆凭证超时时间
*/
int DEFAULT_EXPIRE_SECONDS=3600*12;
/**
* 记住状态的登陆凭证超时时间
*/
int REMMERBER_EXPIRED_SECONDS=3600*12*100;
}
6.随机字符串和加密工具类:
package com.nowcoder.community.util;
import org.apache.commons.lang3.StringUtils;
import org.springframework.util.DigestUtils;
import java.util.UUID;
public class CommunityUtil {
//生成随机字符串
public static String generateUUID()
{
return UUID.randomUUID().toString().replaceAll("-","");
}
//MD5加密
public static String md5(String key){
if (StringUtils.isBlank(key)){
return null;
}
return DigestUtils.md5DigestAsHex(key.getBytes());
}
}