您好,登錄后才能下訂單哦!
原創(chuàng)文章,歡迎轉載。轉載請注明:轉載自IT人故事會,謝謝!
原文鏈接地址:『高級篇』docker之開發(fā)用戶服務EdgeService(13)上一節(jié)開發(fā)了用戶服務,即將開發(fā)的是用戶服務EdgeService,從這個調用關系,可以看到用戶的EdgeService是一個服務的服務,首選調用用戶服務,對用戶信息基本的操作,調用信息服務實現發(fā)送短信,發(fā)送郵件,還需要實現登錄和注冊的功能,并且登錄是一個單點登錄需要支持其他的系統,支持課程的登錄的EdgeService,對他的要求是無狀態(tài)的,還需要集中式的緩存redis。這么多服務集中于一身說明它是一個非常復雜的服務,不過也沒關系,我們從頭到尾把他開發(fā)完成。源碼:https://github.com/limingios/msA-docker
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.3.RELEASE</version>
</parent>
<modelVersion>4.0.0</modelVersion>
<groupId>com.idig8</groupId>
<artifactId>user-edge-service</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.apache.thrift</groupId>
<artifactId>libthrift</artifactId>
<version>0.10.0</version>
</dependency>
<dependency>
<groupId>com.idig8</groupId>
<artifactId>user-thrift-service-api</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>com.idig8</groupId>
<artifactId>message-thrift-service-api</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>commons-lang</groupId>
<artifactId>commons-lang</artifactId>
<version>2.6</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot</artifactId>
<version>RELEASE</version>
<scope>compile</scope>
</dependency>
</dependencies>
</project>
package com.idig8.user.redis;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cache.CacheManager;
import org.springframework.cache.annotation.CachingConfigurerSupport;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
/**
Redis緩存配置類*/
@Configuration
br/>*/
@Configuration
public class RedisConfig extends CachingConfigurerSupport {
@Value("${spring.redis.host}")
private String host;@Value("${spring.redis.port}")
br/>@Value("${spring.redis.port}")
br/>@Value("${spring.redis.timeout}")
br/>@Value("${spring.redis.password}")
//緩存管理器@Bean
br/>@Bean
RedisCacheManager cacheManager = new RedisCacheManager(redisTemplate);
//設置緩存過期時間
cacheManager.setDefaultExpiration(10000);
return cacheManager;
}
@Bean
public JedisConnectionFactory redisConnectionFactory() {
JedisConnectionFactory factory = new JedisConnectionFactory();
factory.setHostName(host);
factory.setPort(port);
factory.setTimeout(timeout);
factory.setPassword(password);
return factory;
}
@Bean
public RedisTemplate<String, String> redisTemplate(RedisConnectionFactory factory){
StringRedisTemplate template = new StringRedisTemplate(factory);
setSerializer(template);//設置序列化工具
template.afterPropertiesSet();
return template;
}
private void setSerializer(StringRedisTemplate template){
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
ObjectMapper om = new ObjectMapper();
om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(om);
template.setValueSerializer(jackson2JsonRedisSerializer);
}
}
RedisClient
``` java
package com.idig8.user.redis;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import java.util.concurrent.TimeUnit;
/**
* Created by liming
*/
@Component
public class RedisClient {
@Autowired
private RedisTemplate redisTemplate;
public <T> T get(String key) {
return (T)redisTemplate.opsForValue().get(key);
}
public void set(String key, Object value) {
redisTemplate.opsForValue().set(key, value);
}
public void set(String key, Object value, int timeout) {
redisTemplate.opsForValue().set(key, value, timeout, TimeUnit.SECONDS);
}
public void expire(String key, int timeout) {
redisTemplate.expire(key, timeout, TimeUnit.SECONDS);
}
}
Response
package com.idig8.user.response;
import java.io.Serializable;
/**
* Created by liming
*/
public class Response implements Serializable {
public static final Response USERNAME_PASSWORD_INVALID = new Response("1001", "username or password invalid");
public static final Response MOBILE_OR_EMAIL_REQUIRED = new Response("1002", "mobile or email is required");
public static final Response SEND_VERIFYCODE_FAILED = new Response("1003", "send verify code failed");
public static final Response VERIFY_CODE_INVALID = new Response("1004", "verifyCode invalid");
public static final Response SUCCESS = new Response();
private String code;
private String message;
public Response() {
this.code = "0";
this.message = "success";
}
public Response(String code, String message) {
this.code = code;
this.message = message;
}
public static Response exception(Exception e) {
return new Response("9999", e.getMessage());
}
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public String getMessage() {
return message;
}
public void setMessage(String message) {
this.message = message;
}
}
LoginResponse
package com.idig8.user.response;
/**
* Created by liming
*/
public class LoginResponse extends Response {
private String token;
public LoginResponse(String token) {
this.token = token;
}
public String getToken() {
return token;
}
public void setToken(String token) {
this.token = token;
}
}
package com.idig8.user.thrift;
import com.idig8.thrift.message.MessageService;
import com.idig8.thrift.user.UserService;
import org.apache.thrift.TServiceClient;
import org.apache.thrift.protocol.TBinaryProtocol;
import org.apache.thrift.protocol.TProtocol;
import org.apache.thrift.transport.*;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
@Component
public class ServiceProvider {
@Value("${thrift.user.ip}")
private String serverIp;
@Value("${thrift.user.port}")
private int serverPort;
@Value("${thrift.message.ip}")
private String messageServerIp;
@Value("${thrift.message.port}")
private int messageServerPort;
private enum ServiceType {
USER,
MESSAGE
}
public UserService.Client getUserService() {
return getService(serverIp, serverPort, ServiceType.USER);
}
public MessageService.Client getMessasgeService() {
return getService(messageServerIp, messageServerPort, ServiceType.MESSAGE);
}
public <T> T getService(String ip, int port, ServiceType serviceType) {
TSocket socket = new TSocket(ip, port, 3000);
TTransport transport = new TFramedTransport(socket);
try {
transport.open();
} catch (TTransportException e) {
e.printStackTrace();
return null;
}
TProtocol protocol = new TBinaryProtocol(transport);
TServiceClient result = null;
switch (serviceType) {
case USER:
result = new UserService.Client(protocol);
break;
case MESSAGE:
result = new MessageService.Client(protocol);
break;
}
return (T)result;
}
}
因為userInfo是通過thrift自動升成的,里面很多方法太過麻煩,不利于開發(fā)查看數據內容,最好的方式自己創(chuàng)建一個對象,將自動升成userInfo轉換成自定義的UserDTO,USerDTO最好的方式是在thrift的工程中進行的。如果多個項目,比較方便,單獨的用戶的edgeservice中進行DTO的話,只能他自己用業(yè)務不清晰。
package com.idig8.user.controller;
import com.idig8.thrift.user.UserInfo;
import com.idig8.thrift.user.dto.UserDTO;
import com.idig8.user.redis.RedisClient;
import com.idig8.user.response.LoginResponse;
import com.idig8.user.response.Response;
import com.idig8.user.thrift.ServiceProvider;
import org.apache.commons.lang.StringUtils;
import org.apache.thrift.TException;
import org.apache.tomcat.util.buf.HexUtils;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;
import java.security.MessageDigest;
import java.util.Random;
@Controller
public class UserController {
@Autowired
private ServiceProvider serviceProvider;
@Autowired
private RedisClient redisClient;
@RequestMapping(value = "/login",method = RequestMethod.POST)
@ResponseBody
public Response login(@RequestParam("username")String username,
@RequestParam("password")String password){
// 1. 驗證用戶名密碼
UserInfo userInfo = null;
try {
userInfo = serviceProvider.getUserService().getUserByName(username);
} catch (TException e) {
e.printStackTrace();
return Response.USERNAME_PASSWORD_INVALID;
}
if (userInfo == null){
return Response.USERNAME_PASSWORD_INVALID;
}
if(!userInfo.getPassword().equalsIgnoreCase(md5(password))){
return Response.USERNAME_PASSWORD_INVALID;
}
// 2. 生成token
String token = genToken();
// 3. 緩存用戶
//因為userInfo是通過thrift自動升成的,里面很多方法太過麻煩,不利于開發(fā)查看數據內容
//最好的方式自己創(chuàng)建一個對象,將自動升成userInfo轉換成自定義的UserDTO
redisClient.set(token,toDTO(userInfo));
return new LoginResponse(token);
}
@RequestMapping(value = "/sendVerifyCode", method = RequestMethod.POST)
@ResponseBody
public Response sendVerifyCode(@RequestParam(value="mobile", required = false) String mobile,
@RequestParam(value="email", required = false) String email) {
String message = "Verify code is:";
String code = randomCode("0123456789", 6);
try {
boolean result = false;
if(StringUtils.isNotBlank(mobile)) {
result = serviceProvider.getMessasgeService().sendMobileMessage(mobile, message+code);
redisClient.set(mobile, code);
} else if(StringUtils.isNotBlank(email)) {
result = serviceProvider.getMessasgeService().sendEmailMessage(email, message+code);
redisClient.set(email, code);
} else {
return Response.MOBILE_OR_EMAIL_REQUIRED;
}
if(!result) {
return Response.SEND_VERIFYCODE_FAILED;
}
} catch (TException e) {
e.printStackTrace();
return Response.exception(e);
}
return Response.SUCCESS;
}
@RequestMapping(value="/register", method = RequestMethod.POST)
@ResponseBody
public Response register(@RequestParam("username") String username,
@RequestParam("password") String password,
@RequestParam(value="mobile", required = false) String mobile,
@RequestParam(value="email", required = false) String email,
@RequestParam("verifyCode") String verifyCode) {
if(StringUtils.isBlank(mobile) && StringUtils.isBlank(email)) {
return Response.MOBILE_OR_EMAIL_REQUIRED;
}
if(StringUtils.isNotBlank(mobile)) {
String redisCode = redisClient.get(mobile);
if(!verifyCode.equals(redisCode)) {
return Response.VERIFY_CODE_INVALID;
}
}else {
String redisCode = redisClient.get(email);
if(!verifyCode.equals(redisCode)) {
return Response.VERIFY_CODE_INVALID;
}
}
UserInfo userInfo = new UserInfo();
userInfo.setUsername(username);
userInfo.setPassword(md5(password));
userInfo.setMobile(mobile);
userInfo.setEmail(email);
try {
serviceProvider.getUserService().regiserUser(userInfo);
} catch (TException e) {
e.printStackTrace();
return Response.exception(e);
}
return Response.SUCCESS;
}
private UserDTO toDTO(UserInfo userInfo) {
UserDTO userDTO = new UserDTO();
BeanUtils.copyProperties(userInfo, userDTO);
return userDTO;
}
private String genToken() {
return randomCode("0123456789abcdefghijklmnopqrstuvwxyz", 32);
}
private String randomCode(String s, int size) {
StringBuilder result = new StringBuilder(size);
Random random = new Random();
for(int i=0;i<size;i++) {
int loc = random.nextInt(s.length());
result.append(s.charAt(loc));
}
return result.toString();
}
private String md5(String password) {
try {
MessageDigest md5 = MessageDigest.getInstance("MD5");
byte[] md5Bytes = md5.digest(password.getBytes("utf-8"));
return HexUtils.toHexString(md5Bytes);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
}
package com.idig8.user.client;
import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.idig8.thrift.user.dto.UserDTO;
import org.apache.commons.lang.StringUtils;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.impl.client.DefaultHttpClient;
import org.codehaus.jackson.map.ObjectMapper;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.util.concurrent.TimeUnit;
/**
* Created by Michael on 2017/10/31.
*/
public abstract class LoginFilter implements Filter {
private static Cache<String, UserDTO> cache =
CacheBuilder.newBuilder().maximumSize(10000)
.expireAfterWrite(3, TimeUnit.MINUTES).build();
public void init(FilterConfig filterConfig) throws ServletException {
}
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest)servletRequest;
HttpServletResponse response = (HttpServletResponse)servletResponse;
String token = request.getParameter("token");
if(StringUtils.isBlank(token)) {
Cookie[] cookies = request.getCookies();
if(cookies!=null) {
for(Cookie c : cookies) {
if(c.getName().equals("token")) {
token = c.getValue();
}
}
}
}
UserDTO userDTO = null;
if(StringUtils.isNotBlank(token)) {
userDTO = cache.getIfPresent(token);
if(userDTO==null) {
userDTO = requestUserInfo(token);
if(userDTO!=null) {
cache.put(token, userDTO);
}
}
}
if(userDTO==null) {
response.sendRedirect("http://www.mooc.com/user/login");
return;
}
login(request, response, userDTO);
filterChain.doFilter(request, response);
}
protected abstract String userEdgeServiceAddr();
protected abstract void login(HttpServletRequest request, HttpServletResponse response, UserDTO userDTO);
private UserDTO requestUserInfo(String token) {
String url = "http://"+userEdgeServiceAddr()+"/user/authentication";
HttpClient client = new DefaultHttpClient();
HttpPost post = new HttpPost(url);
post.addHeader("token", token);
InputStream inputStream = null;
try {
HttpResponse response = client.execute(post);
if(response.getStatusLine().getStatusCode()!= HttpStatus.SC_OK) {
throw new RuntimeException("request user info failed! StatusLine:"+response.getStatusLine());
}
inputStream = response.getEntity().getContent();
byte[] temp = new byte[1024];
StringBuilder sb = new StringBuilder();
int len = 0;
while((len = inputStream.read(temp))>0) {
sb.append(new String(temp,0,len));
}
UserDTO userDTO = new ObjectMapper().readValue(sb.toString(), UserDTO.class);
return userDTO;
} catch (IOException e) {
e.printStackTrace();
} finally {
if(inputStream!=null) {
try{
inputStream.close();
}catch(Exception e) {
e.printStackTrace();
}
}
}
return null;
}
public void destroy() {
}
}
需要引入單點登錄的模塊進行實現的功能
PS:其實通過梳理發(fā)現這個還是有套路可尋的如何多語言進行通信,先生成對應的語言的代碼,然后通過rpc的服務端和客戶端,他們之前進行協議話的通信,服務端完成自身的業(yè)務邏輯,客戶端就獲取返回的結果。
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。