DefaultHttpRequestHandler.java 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445
  1. package org.jim.server.http;
  2. import java.beans.PropertyDescriptor;
  3. import java.io.File;
  4. import java.lang.reflect.Method;
  5. import java.util.HashMap;
  6. import java.util.Map;
  7. import java.util.Map.Entry;
  8. import java.util.Set;
  9. import java.util.concurrent.ExecutionException;
  10. import org.apache.commons.lang3.StringUtils;
  11. import org.slf4j.Logger;
  12. import org.slf4j.LoggerFactory;
  13. import org.tio.core.ChannelContext;
  14. import org.jim.common.http.Cookie;
  15. import org.jim.common.http.HttpConfig;
  16. import org.jim.common.http.HttpConst;
  17. import org.jim.common.http.HttpRequest;
  18. import org.jim.common.http.HttpResponse;
  19. import org.jim.common.http.HttpResponseStatus;
  20. import org.jim.common.http.RequestLine;
  21. import org.jim.common.http.handler.IHttpRequestHandler;
  22. import org.jim.common.http.listener.IHttpServerListener;
  23. import org.jim.common.http.session.HttpSession;
  24. import org.jim.server.http.mvc.Routes;
  25. import org.jim.server.util.ClassUtils;
  26. import org.jim.server.util.HttpResps;
  27. import org.tio.utils.cache.guava.GuavaCache;
  28. import cn.hutool.core.bean.BeanUtil;
  29. import cn.hutool.core.convert.Convert;
  30. import cn.hutool.core.io.FileUtil;
  31. import cn.hutool.core.util.ClassUtil;
  32. /**
  33. *
  34. * @author WChao
  35. *
  36. */
  37. public class DefaultHttpRequestHandler implements IHttpRequestHandler {
  38. private static Logger log = LoggerFactory.getLogger(DefaultHttpRequestHandler.class);
  39. // /**
  40. // * 静态资源的CacheName
  41. // * key: path 譬如"/index.html"
  42. // * value: HttpResponse
  43. // */
  44. // private static final String STATIC_RES_CACHENAME = "TIO_HTTP_STATIC_RES";
  45. /**
  46. * 静态资源的CacheName
  47. * key: path 譬如"/index.html"
  48. * value: FileCache
  49. */
  50. private static final String STATIC_RES_CONTENT_CACHENAME = "TIO_HTTP_STATIC_RES_CONTENT";
  51. /**
  52. * @param args
  53. *
  54. * @author WChao
  55. * 2016年11月18日 上午9:13:15
  56. *
  57. */
  58. public static void main(String[] args) {
  59. }
  60. protected HttpConfig httpConfig;
  61. protected Routes routes = null;
  62. // private LoadingCache<String, HttpSession> loadingCache = null;
  63. private IHttpServerListener httpServerListener;
  64. private GuavaCache staticResCache;
  65. /**
  66. *
  67. * @param httpConfig
  68. * @author WChao
  69. */
  70. public DefaultHttpRequestHandler(HttpConfig httpConfig) {
  71. this.httpConfig = httpConfig;
  72. if (httpConfig.getMaxLiveTimeOfStaticRes() > 0) {
  73. // GuavaCache.register(STATIC_RES_CACHENAME, (long) httpConfig.getMaxLiveTimeOfStaticRes(), null);
  74. staticResCache = GuavaCache.register(STATIC_RES_CONTENT_CACHENAME, (long) httpConfig.getMaxLiveTimeOfStaticRes(), null);
  75. }
  76. this.setHttpServerListener(httpConfig.getHttpServerListener());
  77. }
  78. /**
  79. *
  80. * @param httpConfig
  81. * @param routes
  82. * @author WChao
  83. */
  84. public DefaultHttpRequestHandler(HttpConfig httpConfig, Routes routes) {
  85. this(httpConfig);
  86. this.routes = routes;
  87. }
  88. /**
  89. * 创建httpsession
  90. * @return
  91. * @author WChao
  92. */
  93. private HttpSession createSession() {
  94. String sessionId = httpConfig.getSessionIdGenerator().sessionId(httpConfig);
  95. HttpSession httpSession = new HttpSession(sessionId);
  96. return httpSession;
  97. }
  98. /**
  99. * @return the httpConfig
  100. */
  101. public HttpConfig getHttpConfig() {
  102. return httpConfig;
  103. }
  104. public IHttpServerListener getHttpServerListener() {
  105. return httpServerListener;
  106. }
  107. private Cookie getSessionCookie(HttpRequest request, HttpConfig httpConfig) throws ExecutionException {
  108. Cookie sessionCookie = request.getCookie(httpConfig.getSessionCookieName());
  109. return sessionCookie;
  110. }
  111. /**
  112. * @return the staticResCache
  113. */
  114. public GuavaCache getStaticResCache() {
  115. return staticResCache;
  116. }
  117. @Override
  118. public HttpResponse handler(HttpRequest request, RequestLine requestLine) throws Exception {
  119. log.info("执行 DefaultHttpRequestHandler.handler()方法...");
  120. HttpResponse ret = null;
  121. try {
  122. processCookieBeforeHandler(request, requestLine);
  123. HttpSession httpSession = request.getHttpSession();//(HttpSession) channelContext.getAttribute();
  124. if (httpServerListener != null) {
  125. ret = httpServerListener.doBeforeHandler(request, requestLine, ret);
  126. if (ret != null) {
  127. return ret;
  128. }
  129. }
  130. String path = requestLine.getPath();
  131. String initPath = path;
  132. Method method = routes.pathMethodMap.get(initPath);
  133. if (method != null) {
  134. String[] paramNames = routes.methodParamnameMap.get(method);
  135. Class<?>[] parameterTypes = method.getParameterTypes();
  136. Object bean = routes.methodBeanMap.get(method);
  137. Object obj = null;
  138. Map<String, Object[]> params = request.getParams();
  139. if (parameterTypes == null || parameterTypes.length == 0) {
  140. obj = method.invoke(bean);
  141. } else {
  142. //赋值这段代码待重构,先用上
  143. Object[] paramValues = new Object[parameterTypes.length];
  144. int i = 0;
  145. for (Class<?> paramType : parameterTypes) {
  146. try {
  147. if (paramType.isAssignableFrom(HttpRequest.class)) {
  148. paramValues[i] = request;
  149. } else if (paramType == HttpSession.class) {
  150. paramValues[i] = httpSession;
  151. } else if (paramType.isAssignableFrom(HttpConfig.class)) {
  152. paramValues[i] = httpConfig;
  153. } else if (paramType.isAssignableFrom(ChannelContext.class)) {
  154. paramValues[i] = request.getChannelContext();
  155. } else {
  156. if (params != null) {
  157. if (ClassUtils.isSimpleTypeOrArray(paramType)) {
  158. Object[] value = params.get(paramNames[i]);
  159. if (value != null && value.length > 0) {
  160. if (paramType.isArray()) {
  161. paramValues[i] = Convert.convert(paramType, value);
  162. } else {
  163. paramValues[i] = Convert.convert(paramType, value[0]);
  164. }
  165. }
  166. } else {
  167. paramValues[i] = paramType.newInstance();//BeanUtil.mapToBean(params, paramType, true);
  168. Set<Entry<String, Object[]>> set = params.entrySet();
  169. label2: for (Entry<String, Object[]> entry : set) {
  170. String fieldName = entry.getKey();
  171. Object[] fieldValue = entry.getValue();
  172. PropertyDescriptor propertyDescriptor = BeanUtil.getPropertyDescriptor(paramType, fieldName, true);
  173. if (propertyDescriptor == null) {
  174. continue label2;
  175. } else {
  176. Method writeMethod = propertyDescriptor.getWriteMethod();
  177. if (writeMethod == null) {
  178. continue label2;
  179. }
  180. writeMethod = ClassUtil.setAccessible(writeMethod);
  181. Class<?>[] clazzes = writeMethod.getParameterTypes();
  182. if (clazzes == null || clazzes.length != 1) {
  183. log.info("方法的参数长度不为1,{}.{}", paramType.getName(), writeMethod.getName());
  184. continue label2;
  185. }
  186. Class<?> clazz = clazzes[0];
  187. if (ClassUtils.isSimpleTypeOrArray(clazz)) {
  188. if (fieldValue != null && fieldValue.length > 0) {
  189. if (clazz.isArray()) {
  190. writeMethod.invoke(paramValues[i], Convert.convert(clazz, fieldValue));
  191. } else {
  192. writeMethod.invoke(paramValues[i], Convert.convert(clazz, fieldValue[0]));
  193. }
  194. }
  195. }
  196. }
  197. }
  198. }
  199. }
  200. }
  201. } catch (Exception e) {
  202. log.error(e.toString(), e);
  203. } finally {
  204. i++;
  205. }
  206. }
  207. obj = method.invoke(bean, paramValues);
  208. }
  209. if (obj instanceof HttpResponse) {
  210. ret = (HttpResponse) obj;
  211. return ret;
  212. } else {
  213. throw new Exception(bean.getClass().getName() + "#" + method.getName() + "返回的对象不是" + HttpResponse.class.getName());
  214. }
  215. } else {
  216. GuavaCache contentCache = null;
  217. FileCache fileCache = null;
  218. if (httpConfig.getMaxLiveTimeOfStaticRes() > 0) {
  219. contentCache = GuavaCache.getCache(STATIC_RES_CONTENT_CACHENAME);
  220. fileCache = (FileCache) contentCache.get(initPath);
  221. }
  222. if (fileCache != null) {
  223. byte[] bodyBytes = fileCache.getData();
  224. Map<String, String> headers = fileCache.getHeaders();
  225. long lastModified = fileCache.getLastModified();
  226. log.info("从缓存获取:[{}], {}", path, bodyBytes.length);
  227. ret = HttpResps.try304(request, lastModified);
  228. if (ret != null) {
  229. ret.addHeader(HttpConst.ResponseHeaderKey.tio_from_cache, "true");
  230. return ret;
  231. }
  232. ret = new HttpResponse(request, httpConfig);
  233. ret.setBody(bodyBytes, request);
  234. ret.addHeaders(headers);
  235. return ret;
  236. } else {
  237. String root = FileUtil.getAbsolutePath(httpConfig.getPageRoot());
  238. File file = new File(root + path);
  239. if (!file.exists() || file.isDirectory()) {
  240. if (StringUtils.endsWith(path, "/")) {
  241. path = path + "index.html";
  242. } else {
  243. path = path + "/index.html";
  244. }
  245. file = new File(root, path);
  246. }
  247. if (file.exists()) {
  248. ret = HttpResps.file(request, file);
  249. ret.setStaticRes(true);
  250. if (contentCache != null && request.getIsSupportGzip()) {
  251. if (ret.getBody() != null && ret.getStatus() == HttpResponseStatus.C200) {
  252. String contentType = ret.getHeader(HttpConst.ResponseHeaderKey.Content_Type);
  253. String contentEncoding = ret.getHeader(HttpConst.ResponseHeaderKey.Content_Encoding);
  254. String lastModified = ret.getHeader(HttpConst.ResponseHeaderKey.Last_Modified);
  255. Map<String, String> headers = new HashMap<>();
  256. if (StringUtils.isNotBlank(contentType)) {
  257. headers.put(HttpConst.ResponseHeaderKey.Content_Type, contentType);
  258. }
  259. if (StringUtils.isNotBlank(contentEncoding)) {
  260. headers.put(HttpConst.ResponseHeaderKey.Content_Encoding, contentEncoding);
  261. }
  262. if (StringUtils.isNotBlank(lastModified)) {
  263. headers.put(HttpConst.ResponseHeaderKey.Last_Modified, lastModified);
  264. }
  265. headers.put(HttpConst.ResponseHeaderKey.tio_from_cache, "true");
  266. fileCache = new FileCache(headers, file.lastModified(), ret.getBody());
  267. contentCache.put(initPath, fileCache);
  268. log.info("放入缓存:[{}], {}", initPath, ret.getBody().length);
  269. }
  270. }
  271. return ret;
  272. }
  273. }
  274. }
  275. ret = resp404(request, requestLine);//Resps.html(request, "404--并没有找到你想要的内容", httpConfig.getCharset());
  276. return ret;
  277. } catch (Exception e) {
  278. logError(request, requestLine, e);
  279. ret = resp500(request, requestLine, e);//Resps.html(request, "500--服务器出了点故障", httpConfig.getCharset());
  280. return ret;
  281. } finally {
  282. if (ret != null) {
  283. try {
  284. processCookieAfterHandler(request, requestLine, ret);
  285. if (httpServerListener != null) {
  286. httpServerListener.doAfterHandler(request, requestLine, ret);
  287. }
  288. } catch (Exception e) {
  289. logError(request, requestLine, e);
  290. }
  291. }
  292. }
  293. }
  294. private void logError(HttpRequest request, RequestLine requestLine, Exception e) {
  295. StringBuilder sb = new StringBuilder();
  296. sb.append("\r\n").append("remote :").append(request.getRemote());
  297. sb.append("\r\n").append("request :").append(requestLine.getLine());
  298. log.error(sb.toString(), e);
  299. }
  300. private void processCookieAfterHandler(HttpRequest request, RequestLine requestLine, HttpResponse httpResponse) throws ExecutionException {
  301. HttpSession httpSession = request.getHttpSession();//(HttpSession) channelContext.getAttribute();//.getHttpSession();//not null
  302. Cookie cookie = getSessionCookie(request, httpConfig);
  303. String sessionId = null;
  304. if (cookie == null) {
  305. String domain = request.getHeader(HttpConst.RequestHeaderKey.Host);
  306. String name = httpConfig.getSessionCookieName();
  307. long maxAge = httpConfig.getSessionTimeout();
  308. // maxAge = Integer.MAX_VALUE; //把过期时间掌握在服务器端
  309. sessionId = httpSession.getId();//randomCookieValue();
  310. cookie = new Cookie(domain, name, sessionId, maxAge);
  311. httpResponse.addCookie(cookie);
  312. httpConfig.getSessionStore().put(sessionId, httpSession);
  313. log.info("{} 创建会话Cookie, {}", request.getChannelContext(), cookie);
  314. } else {
  315. sessionId = cookie.getValue();
  316. HttpSession httpSession1 = (HttpSession) httpConfig.getSessionStore().get(sessionId);
  317. if (httpSession1 == null) {//有cookie但是超时了
  318. sessionId = httpSession.getId();
  319. String domain = request.getHeader(HttpConst.RequestHeaderKey.Host);
  320. String name = httpConfig.getSessionCookieName();
  321. long maxAge = httpConfig.getSessionTimeout();
  322. // maxAge = Long.MAX_VALUE; //把过期时间掌握在服务器端
  323. cookie = new Cookie(domain, name, sessionId, maxAge);
  324. httpResponse.addCookie(cookie);
  325. httpConfig.getSessionStore().put(sessionId, httpSession);
  326. }
  327. }
  328. }
  329. private void processCookieBeforeHandler(HttpRequest request, RequestLine requestLine) throws ExecutionException {
  330. Cookie cookie = getSessionCookie(request, httpConfig);
  331. HttpSession httpSession = null;
  332. if (cookie == null) {
  333. httpSession = createSession();
  334. } else {
  335. String sessionId = cookie.getValue();
  336. httpSession = (HttpSession) httpConfig.getSessionStore().get(sessionId);
  337. if (httpSession == null) {
  338. log.info("{} session【{}】超时", request.getChannelContext(), sessionId);
  339. httpSession = createSession();
  340. }
  341. }
  342. request.setHttpSession(httpSession);
  343. }
  344. @Override
  345. public HttpResponse resp404(HttpRequest request, RequestLine requestLine) {
  346. String file404 = httpConfig.getPage404();
  347. String root = FileUtil.getAbsolutePath(httpConfig.getPageRoot());
  348. File file = new File(root + file404);
  349. if (file.exists()) {
  350. HttpResponse ret = HttpResps.redirect(request, file404 + "?tio_initpath=" + requestLine.getPathAndQuery());
  351. return ret;
  352. } else {
  353. HttpResponse ret = HttpResps.html(request, "404");
  354. return ret;
  355. }
  356. }
  357. @Override
  358. public HttpResponse resp500(HttpRequest request, RequestLine requestLine, Throwable throwable) {
  359. String file500 = httpConfig.getPage500();
  360. String root = FileUtil.getAbsolutePath(httpConfig.getPageRoot());
  361. File file = new File(root + file500);
  362. if (file.exists()) {
  363. HttpResponse ret = HttpResps.redirect(request, file500 + "?tio_initpath=" + requestLine.getPathAndQuery());
  364. return ret;
  365. } else {
  366. HttpResponse ret = HttpResps.html(request, "500");
  367. return ret;
  368. }
  369. }
  370. /**
  371. * @param httpConfig the httpConfig to set
  372. */
  373. public void setHttpConfig(HttpConfig httpConfig) {
  374. this.httpConfig = httpConfig;
  375. }
  376. public void setHttpServerListener(IHttpServerListener httpServerListener) {
  377. this.httpServerListener = httpServerListener;
  378. }
  379. /**
  380. * @param staticResCache the staticResCache to set
  381. */
  382. public void setStaticResCache(GuavaCache staticResCache) {
  383. this.staticResCache = staticResCache;
  384. }
  385. @Override
  386. public void clearStaticResCache(HttpRequest request) {
  387. if (staticResCache != null) {
  388. staticResCache.clear();
  389. }
  390. }
  391. }