springboot是一个scala编写的rest接口交互,scala的连接是akka
1.503 service unbelievable
解:akka-http-core设置request-timeout = 20 s,太短了。httpclient虽然设置6分钟,时间到了。那边服务不可用
2.Connection reset:连接重试,httpclient本来设置超时是6分钟,每次到一分钟左右就出现这个问题
原因:在 akka-http-core里面,linger-timeout = 1 min,一分钟就超时
解1.linger-timeout设置大一点,最好大于超时时间request-timeout,idle-timeout设置大一点,这里非常重要,
解2.改用http 1.0版本,它是一个短链接,关闭连接,同时也改nginx的配置,超时长一点。idle-timeout设置大一点,这里非常重要,
代码示例
nginx.conf
upstream test.local {
server 192.168.1.221:26181;
}
server {
listen 3000;
charset utf-8;
gzip on;
gzip_types application/javascript text/html text/css;
client_max_body_size 250m;
#access_log /var/log/nginx/log/host.access.log main;
location / {
root /home/centos/jiangwancheng/visual/dist;
try_files $uri /index.html;
}
location /redpeak {
#rewrite ^/hs-api/(.*)$ /$1 break;
proxy_pass http://test.local;
proxy_connect_timeout 1800s;
proxy_send_timeout 1800s;
proxy_read_timeout 1800s;
}
#error_page 404 /static/html/404/404.html;
}
import com.fasterxml.jackson.core.type.TypeReference;
import com.hongshan.visual.common.constant.BdpConstant;
import com.hongshan.visual.common.constant.Tips;
import com.hongshan.visual.common.exception.VisualException;
import com.hongshan.visual.common.model.dto.OperateResultDTO;
import com.hongshan.visual.common.model.dto.TenantDTO;
import org.apache.commons.lang3.StringUtils;
import org.apache.http.*;
import org.apache.http.client.HttpClient;
import org.apache.http.client.HttpRequestRetryHandler;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.protocol.HTTP;
import org.apache.http.protocol.HttpContext;
import org.apache.http.util.EntityUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import org.springframework.web.client.RestTemplate;
import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;
import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static com.hongshan.visual.common.constant.BdpConstant.TENANT_ID;
/**
* http相关的工具类
*/
@Component
public class HttpUtil {
private static final Logger log = LoggerFactory.getLogger(HttpUtil.class);
public static String platformUrl;
@Value("${platform.url}")
public void setPlatformUrl(String url) {
platformUrl = url;
}
/**
* 封装url
*
* @param domain 域名
* @param request 请求
* @param params 参数
*
* @return 封装完毕的url
*/
private static String makeUrl(String domain, String request, Map<String, String> params) {
StringBuilder sb = new StringBuilder().append(domain).append(request);
if (!params.isEmpty()) {
sb.append("?");
for (Map.Entry entry : params.entrySet()) {
sb.append(entry.getKey()).append("=").append(entry.getValue()).append("&");
}
sb.delete(sb.length() - 1, sb.length());
}
return sb.toString();
}
/**
* get请求
*
* @param domain 域名
* @param request 请求
* @param params 参数
*/
public static Map<String, String> doGet(String domain, String request, Map<String, String> params) throws Exception {
Map<String, String> res = new HashMap<String, String>();
HttpURLConnection connection = null;
BufferedReader br = null;
try {
String urlStr = makeUrl(domain, request, params);
URL url = new URL(urlStr);
connection = (HttpURLConnection) url.openConnection();
connection.setDoOutput(true); // 设置该连接是可以输出的
connection.setRequestMethod("GET"); // 设置请求方式
connection.setRequestProperty("Content-Type", "application/x-www-form-urlencoded; charset=UTF-8");
connection.connect();
int code = connection.getResponseCode();
res.put("code", code + "");
if (code == 200) {
br = new BufferedReader(new InputStreamReader(connection.getInputStream(), "utf-8"));
StringBuilder result = new StringBuilder();
String line;
while ((line = br.readLine()) != null) { // 读取数据
result.append(line).append(System.getProperty("line.separator"));
}
res.put("message", result.toString());
}else {
res.put("message", connection.getResponseMessage());
}
} catch (Exception e) {
throw new Exception(e);
} finally {
if (connection != null) {
connection.disconnect();
}
if (br != null) {
br.close();
}
}
return res;
}
public static String getRestTemplateResult(String uri,String paramName,String jsonStr){
RestTemplate restTemplate = SpringUtil.getRestTemplate();
String url = platformUrl + uri + "?" + paramName + "=" + jsonStr;
try {
return restTemplate.getForObject(url, String.class);
}catch (Exception e){
log.error(Tips.DBP_EXCEPTION + "method:getRestTemplateResult, errMsg:{}",e);
throw new VisualException(Tips.DBP_EXCEPTION);
}
}
public static String getRestPlatform(String uri,String parmasStr){
RestTemplate restTemplate = SpringUtil.getRestTemplate();
String url = platformUrl + uri + "?" + parmasStr;
if(StringUtils.isEmpty(parmasStr)){
url = platformUrl + uri;
}
try {
return restTemplate.getForObject(url, String.class);
}catch (Exception e){
log.error(Tips.DBP_EXCEPTION + " method:getResultByDay,errMsg:{}",e);
throw new VisualException(Tips.DBP_EXCEPTION);
}
}
/**
* 向目的URL发送post请求
* @param uri 目的uri
* @param jsonString 发送的json参数
* @return ResultVO
*/
public static String postRest(String uri, String jsonString) throws IOException {
Map<String, String> httpResMap;
try{
httpResMap = doPost(platformUrl, uri, jsonString);
}catch (Exception e){
log.error(Tips.DBP_EXCEPTION + " method:postRest,errMsg:{}",e);
throw new VisualException(Tips.DBP_EXCEPTION);
}
if (! httpResMap.get(BdpConstant.CODE_STRING).equals(BdpConstant.SUCCESS_CODE)){
log.error(Tips.DBP_EXCEPTION );
throw new VisualException(httpResMap.get(BdpConstant.CODE_STRING) + ":" +httpResMap.get("message"));
}
return httpResMap.get("message");
}
public static Map<String, String> doPost(String domain, String request, String jsonStr) throws Exception {
Map<String, String> res = new HashMap<String, String>();
HttpURLConnection connection = null;
BufferedReader br = null;
OutputStream os = null;
try {
String urlStr = domain + request;
URL url = new URL(urlStr);
connection = (HttpURLConnection) url.openConnection();
connection.setDoOutput(true); // 设置该连接是可以输出的
connection.setUseCaches(false);
connection.setRequestMethod("POST"); // 设置请求方式
connection.setRequestProperty("Content-Type", "application/json; charset=UTF-8");
connection.setRequestProperty("Cache-Control", "no-cache");
connection.setRequestProperty("Accept-Encoding", "gzip, deflate,compress");
connection.connect();
connection.setConnectTimeout(6*60*BdpConstant.BASE_TIME_OUT);
connection.setReadTimeout(6*60*BdpConstant.BASE_TIME_OUT);
os = connection.getOutputStream();
os.write(jsonStr.getBytes("UTF-8"));
os.flush();
int code = connection.getResponseCode();
res.put("code", code + "");
if (code == 200) {
br = new BufferedReader(new InputStreamReader(connection.getInputStream(), "utf-8"));
StringBuilder result = new StringBuilder();
String line;
while ((line = br.readLine()) != null) { // 读取数据
result.append(line).append(System.getProperty("line.separator"));
}
res.put("message", result.toString());
return res;
}else {
res.put("message", connection.getResponseMessage());
}
} catch (Exception e) {
throw new Exception(e);
} finally {
if (connection != null) {
connection.disconnect();
}
if (br != null) {
br.close();
}
if (os != null) {
os.close();
}
}
return res;
}
/**
* 向目的URL发送post请求
* @param uri 目的uri
* @param jsonStr 发送的json参数
* @param readTimeOut 超时
* @return OperateResultDTO 操作返回
*/
public static OperateResultDTO debugPost(String uri, String jsonStr, int readTimeOut){
OperateResultDTO dto ;
HttpURLConnection connection = null;
BufferedReader br = null;
OutputStream os = null;
try {
URL url = new URL(uri);
connection = (HttpURLConnection) url.openConnection();
connection.setDoOutput(true);
connection.setUseCaches(false);
connection.setRequestMethod("POST");
connection.setRequestProperty("Content-Type", "application/json; charset=UTF-8");
connection.setRequestProperty("Cache-Control", "no-cache");
connection.setRequestProperty("Accept-Encoding", "gzip, deflate,compress");
connection.connect();
connection.setConnectTimeout(2000);
connection.setReadTimeout(readTimeOut);
os = connection.getOutputStream();
os.write(jsonStr.getBytes(StandardCharsets.UTF_8));
os.flush();
int code = connection.getResponseCode();
if (code == 200) {
br = new BufferedReader(new InputStreamReader(connection.getInputStream(), StandardCharsets.UTF_8));
StringBuilder result = new StringBuilder();
String line;
while ((line = br.readLine()) != null) {
result.append(line).append(System.getProperty("line.separator"));
}
dto = JsonResult.MAPPER.readValue(result.toString(), new TypeReference<OperateResultDTO>() {});
return dto;
}else {
dto = new OperateResultDTO(null,1,connection.getResponseMessage());
}
} catch (IOException e) {
log.error(e.getMessage());
dto = new OperateResultDTO(null,1,e.getMessage());
} finally {
if (connection != null) {
connection.disconnect();
}
close(br);
close(os);
}
return dto;
}
public static void close(AutoCloseable stream){
if (stream != null) {
try {
stream.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
/**
* @Description: 获取租户信息
* @Param:
* @Author: luoliyin
* @Date: 2020/4/22
**/
public static List<TenantDTO> getTenant(){
try {
String json1 = getRestPlatform(TENANT_ID, null);
return JsonResult.MAPPER.readValue(json1, new TypeReference<List<TenantDTO>>() {});
} catch (Exception e) {
log.error("获取租户,连接平台异常!");
throw new VisualException("平台正在重启中,请稍后再试!");
}
}
private static HttpRequestRetryHandler myRetryHandler = new HttpRequestRetryHandler() {
@Override
public boolean retryRequest(IOException exception,
int executionCount, HttpContext context) {
return false;
}};
/**
* post请求(用于请求json格式的参数)
* @param url
* @param params
* @return
*/
public static String doPost(String url, String params) throws Exception {
HttpClientBuilder custom = HttpClients.custom();
CloseableHttpClient httpclient = custom.setRetryHandler(myRetryHandler).build();
HttpPost httpPost = new HttpPost(platformUrl + url);
httpPost.setHeader("Accept", "application/json");
httpPost.setHeader("Content-Type", "application/json");
String charSet = "UTF-8";
StringEntity entity = new StringEntity(params, charSet);
httpPost.setEntity(entity);
RequestConfig requestConfig = RequestConfig.custom()
.setConnectTimeout(30*60*BdpConstant.BASE_TIME_OUT)
.setSocketTimeout(30*60*BdpConstant.BASE_TIME_OUT)
.setConnectionRequestTimeout(30*60*BdpConstant.BASE_TIME_OUT)
.build();
httpPost.setProtocolVersion(HttpVersion.HTTP_1_0);
httpPost.addHeader(HTTP.CONN_DIRECTIVE, HTTP.CONN_CLOSE);
httpPost.setConfig(requestConfig);
CloseableHttpResponse response = null;
try {
response = httpclient.execute(httpPost);
StatusLine status = response.getStatusLine();
int state = status.getStatusCode();
if (state == HttpStatus.SC_OK) {
HttpEntity responseEntity = response.getEntity();
String jsonString = EntityUtils.toString(responseEntity);
return jsonString;
}
else{
log.error("请求返回:"+state+"(" + url + ")" );
throw new VisualException("请求平台异常:"+state);
}
}
finally {
if (response != null) {
try {
response.close();
} catch (IOException e) {
e.printStackTrace();
}
}
try {
httpclient.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}