当前位置: 首页 > news >正文

郑州做网站推广地指数平滑法

郑州做网站推广地,指数平滑法,wordpress 主题原理,上虞市建设风机厂网站在基于SpringCloud开发的微服务中#xff0c;我们一般会选择在网关层记录请求和响应日志#xff0c;并将其收集到ELK中用作查询和分析。 今天我们就来看看如何实现此功能。 日志实体类 首先我们在网关中定义一个日志实体#xff0c;用于组装日志对象 Data public class …在基于SpringCloud开发的微服务中我们一般会选择在网关层记录请求和响应日志并将其收集到ELK中用作查询和分析。 今天我们就来看看如何实现此功能。 日志实体类 首先我们在网关中定义一个日志实体用于组装日志对象 Data public class AccessLog {/**用户编号**/private Long userId;/**路由**/private String targetServer;/**协议**/private String schema;/**请求方法名**/private String requestMethod;/**访问地址**/private String requestUrl;/**请求IP**/private String clientIp;/**查询参数**/private MultiValueMapString, String queryParams;/**请求体**/private String requestBody;/**请求头**/private MultiValueMapString, String requestHeaders;/**响应体**/private String responseBody;/**响应头**/private MultiValueMapString, String responseHeaders;/**响应结果**/private HttpStatusCode httpStatusCode;/**开始请求时间**/private LocalDateTime startTime;/**结束请求时间**/private LocalDateTime endTime;/**执行时长单位毫秒**/private Integer duration;}网关日志过滤器 接下来我们在网关中定义一个Filter用于收集日志信息。 Component public class AccessLogFilter implements GlobalFilter, Ordered {private final ListHttpMessageReader? messageReaders  HandlerStrategies.withDefaults().messageReaders();/*** 打印日志* param accessLog 网关日志*/private void writeAccessLog(AccessLog accessLog) {log.info(----access---- : {}, JsonUtils.obj2StringPretty(accessLog));}/*** 顺序必须是-1否则标准的NettyWriteResponseFilter将在您的过滤器得到一个被调用的机会之前发送响应* 也就是说如果不小于 -1 将不会执行获取后端响应的逻辑* return*/Overridepublic int getOrder() {return -100;}Overridepublic MonoVoid filter(ServerWebExchange exchange, GatewayFilterChain chain) {// 将 Request 中可以直接获取到的参数设置到网关日志ServerHttpRequest request  exchange.getRequest();AccessLog gatewayLog  new AccessLog();gatewayLog.setTargetServer(WebUtils.getGatewayRoute(exchange).getId());gatewayLog.setSchema(request.getURI().getScheme());gatewayLog.setRequestMethod(request.getMethod().name());gatewayLog.setRequestUrl(request.getURI().getRawPath());gatewayLog.setQueryParams(request.getQueryParams());gatewayLog.setRequestHeaders(request.getHeaders());gatewayLog.setStartTime(LocalDateTime.now());gatewayLog.setClientIp(WebUtils.getClientIP(exchange));// 继续 filter 过滤MediaType mediaType  request.getHeaders().getContentType();if (MediaType.APPLICATION_FORM_URLENCODED.isCompatibleWith(mediaType)|| MediaType.APPLICATION_JSON.isCompatibleWith(mediaType)) { // 适合 JSON 和 Form 提交的请求return filterWithRequestBody(exchange, chain, gatewayLog);}return filterWithoutRequestBody(exchange, chain, gatewayLog);}/*** 没有请求体的请求只需要记录日志*/private MonoVoid filterWithoutRequestBody(ServerWebExchange exchange, GatewayFilterChain chain, AccessLog accessLog) {// 包装 Response用于记录 Response BodyServerHttpResponseDecorator decoratedResponse  recordResponseLog(exchange, accessLog);return chain.filter(exchange.mutate().response(decoratedResponse).build()).then(Mono.fromRunnable(() - writeAccessLog(accessLog)));}/*** 需要读取请求体* 参考 {link ModifyRequestBodyGatewayFilterFactory} 实现*/private MonoVoid filterWithRequestBody(ServerWebExchange exchange, GatewayFilterChain chain, AccessLog gatewayLog) {// 设置 Request Body 读取时设置到网关日志ServerRequest serverRequest  ServerRequest.create(exchange, messageReaders);MonoString modifiedBody  serverRequest.bodyToMono(String.class).flatMap(body - {gatewayLog.setRequestBody(body);return Mono.just(body);});// 通过 BodyInserter 插入 body(支持修改body), 避免 request body 只能获取一次BodyInserterMonoString, ReactiveHttpOutputMessage bodyInserter  BodyInserters.fromPublisher(modifiedBody, String.class);HttpHeaders headers  new HttpHeaders();headers.putAll(exchange.getRequest().getHeaders());// the new content type will be computed by bodyInserter// and then set in the request decoratorheaders.remove(HttpHeaders.CONTENT_LENGTH);CachedBodyOutputMessage outputMessage  new CachedBodyOutputMessage(exchange, headers);// 通过 BodyInserter 将 Request Body 写入到 CachedBodyOutputMessage 中return bodyInserter.insert(outputMessage, new BodyInserterContext()).then(Mono.defer(() - {// 重新封装请求ServerHttpRequest decoratedRequest  requestDecorate(exchange, headers, outputMessage);// 记录响应日志ServerHttpResponseDecorator decoratedResponse  recordResponseLog(exchange, gatewayLog);// 记录普通的return chain.filter(exchange.mutate().request(decoratedRequest).response(decoratedResponse).build()).then(Mono.fromRunnable(() - writeAccessLog(gatewayLog))); // 打印日志}));}/*** 记录响应日志* 通过 DataBufferFactory 解决响应体分段传输问题。*/private ServerHttpResponseDecorator recordResponseLog(ServerWebExchange exchange, AccessLog accessLog) {ServerHttpResponse response  exchange.getResponse();return new ServerHttpResponseDecorator(response) {Overridepublic MonoVoid writeWith(Publisher? extends DataBuffer body) {if (body instanceof Flux) {DataBufferFactory bufferFactory  response.bufferFactory();// 计算执行时间accessLog.setEndTime(LocalDateTime.now());accessLog.setDuration((int) (LocalDateTimeUtil.between(accessLog.getStartTime(),accessLog.getEndTime()).toMillis()));accessLog.setResponseHeaders(response.getHeaders());accessLog.setHttpStatusCode(response.getStatusCode());// 获取响应类型如果是 json 就打印String originalResponseContentType  exchange.getAttribute(ServerWebExchangeUtils.ORIGINAL_RESPONSE_CONTENT_TYPE_ATTR);if (StrUtil.isNotBlank(originalResponseContentType) originalResponseContentType.contains(application/json)) {Flux? extends DataBuffer fluxBody  Flux.from(body);return super.writeWith(fluxBody.buffer().map(dataBuffers - {// 设置 response body 到网关日志byte[] content  readContent(dataBuffers);String responseResult  new String(content, StandardCharsets.UTF_8);accessLog.setResponseBody(responseResult);// 响应return bufferFactory.wrap(content);}));}}// if body is not a flux. never got there.return super.writeWith(body);}};}/*** 请求装饰器支持重新计算 headers、body 缓存** param exchange 请求* param headers 请求头* param outputMessage body 缓存* return 请求装饰器*/private ServerHttpRequestDecorator requestDecorate(ServerWebExchange exchange, HttpHeaders headers, CachedBodyOutputMessage outputMessage) {return new ServerHttpRequestDecorator(exchange.getRequest()) {Overridepublic HttpHeaders getHeaders() {long contentLength  headers.getContentLength();HttpHeaders httpHeaders  new HttpHeaders();httpHeaders.putAll(super.getHeaders());if (contentLength  0) {httpHeaders.setContentLength(contentLength);} else {// TODO: this causes a HTTP/1.1 411 Length Required // on// httpbin.orghttpHeaders.set(HttpHeaders.TRANSFER_ENCODING, chunked);}return httpHeaders;}Overridepublic FluxDataBuffer getBody() {return outputMessage.getBody();}};}/*** 从dataBuffers中读取数据* author jam* date 2024/5/26 22:31*/private byte[] readContent(List? extends DataBuffer dataBuffers) {// 合并多个流集合解决返回体分段传输DataBufferFactory dataBufferFactory  new DefaultDataBufferFactory();DataBuffer join  dataBufferFactory.join(dataBuffers);byte[] content  new byte[join.readableByteCount()];join.read(content);// 释放掉内存DataBufferUtils.release(join);return content;}}代码较长建议直接拷贝到编辑器只要注意下面一个关键点 getOrder()方法返回的值必须要-1否则标准的NettyWriteResponseFilter将在您的过滤器被调用的机会之前发送响应即不会执行获取后端响应参数的方法 通过上面的两步我们已经可以获取到请求的输入输出参数了在 writeAccessLog()中将其打印到日志文件方便通过ELK进行收集。 在实际项目中网关日志量一般会非常大不建议使用数据库进行存储。 实际效果 服务正常响应 服务异常响应
http://www.dnsts.com.cn/news/26783.html

相关文章:

  • 营销型网站建设的五力原则整合营销策略
  • 做改网站长春网站长春网络推广建设
  • 网站后台管理系统栏目位置造价咨询公司加盟分公司
  • 网站如何做m适配郑州好的seo外包公司
  • 旅游网站建设的论文相亲网站的女人 做直播的
  • 上海监理建设协会网站番禺网站制作技术
  • 专业的网站建设公个人电脑做网站
  • 购物网站设计目的泰州公司网站建设
  • 建设部网站江苏金安杭州定制网站开发
  • 网站可以做腾讯广告联盟郑州网络公司现状
  • 机关单位建设网站 说明什么是网店
  • 做酒类直供网站行吗网站是怎样制作的
  • 网站建设市场行情分析短视频营销推广
  • 做微景观的网站怎样做农村电商网站
  • 1空间做2个网站吗wordpress修改404页面
  • 英文网站正在建设页面永州网站建设企业
  • 建设招标项目常挂网站有哪些哪个网站开发小程序
  • 数据库与网站建设做公众号的网站有哪些
  • 营销型网站.网站运维主要做些什么工作
  • 网站建设百度推广长宁网站建设价格
  • 学校类网站建设的分析网络推广有几种方法
  • 电子商务网站建设与管理答案wordpress最强的教育网站
  • 合肥网站建设网站模板主机做网站
  • 做棋牌推广网站违反不如何做网站源码备份
  • 太原流量大的网站杭州网站建设维护
  • 惠州网站建设学校seo优化易下拉排名
  • 杭州手机模板建站温州公司网站开发
  • 网上服装商城网站建设方案深圳网站优化课程哪里学
  • 网站建设选用平台分析网站关键词排名外包
  • 建网站英语一起爱又一个wordpress站点