Spring Boot에서 Rest Web Service에 걸린 시간을 기록하는 방법
Spring Boot을 사용하여 Web Rest 웹 서비스를 작성 중입니다.
웹 서비스가 요청을 처리하는 데 걸린 시간을 기록하려고 합니다.또한 호출된 헤더, 메서드 및 URI를 로그에 기록합니다.
나는 몇 달 전에 내 저지 웹 서비스에서도 비슷한 일을 했다.ContainerRequestFilter and ContainerResponseFilter filter() method.
또한 AOP가 더 나은가 아니면 필터인가?
이런 기본 필터 써봤어요?
import java.io.IOException;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
@Component
@WebFilter("/*")
public class StatsFilter implements Filter {
private static final Logger LOGGER = LoggerFactory.getLogger(StatsFilter.class);
@Override
public void init(FilterConfig filterConfig) throws ServletException {
// empty
}
@Override
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws IOException, ServletException {
long time = System.currentTimeMillis();
try {
chain.doFilter(req, resp);
} finally {
time = System.currentTimeMillis() - time;
LOGGER.trace("{}: {} ms ", ((HttpServletRequest) req).getRequestURI(), time);
}
}
@Override
public void destroy() {
// empty
}
}
편집자: (@suren-aznauryan에게 감사함) 이제, 이것은Instant
그리고.Duration
회피하다System.currentTimeMillis()
.
import java.io.IOException;
import java.time.Duration;
import java.time.Instant;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
@Component
@WebFilter("/*")
public class StatsFilter implements Filter {
private static final Logger LOGGER = LoggerFactory.getLogger(StatsFilter.class);
@Override
public void init(FilterConfig filterConfig) throws ServletException {
// empty
}
@Override
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws IOException, ServletException {
Instant start = Instant.now();
try {
chain.doFilter(req, resp);
} finally {
Instant finish = Instant.now();
long time = Duration.between(start, finish).toMillis();
LOGGER.trace("{}: {} ms ", ((HttpServletRequest) req).getRequestURI(), time);
}
}
@Override
public void destroy() {
// empty
}
}
스프링 부트: 로깅 가로채기
public class ApiLogger extends HandlerInterceptorAdapter {
private static final Logger logger = LoggerFactory
.getLogger(ApiLogger.class);
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
String requestId = UUID.randomUUID().toString();
log(request,response, requestId);
long startTime = System.currentTimeMillis();
request.setAttribute("startTime", startTime);
request.setAttribute("requestId", requestId);
return true;
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
super.afterCompletion(request, response, handler, ex);
long startTime = (Long)request.getAttribute("startTime");
long endTime = System.currentTimeMillis();
long executeTime = endTime - startTime;
logger.info("requestId {}, Handle :{} , request take time: {}",request.getAttribute("requestId"), handler, executeTime);
}
private void log(HttpServletRequest request, HttpServletResponse response, String requestId) {
logger.info("requestId {}, host {} HttpMethod: {}, URI : {}",requestId, request.getHeader("host"),
request.getMethod(), request.getRequestURI() );
}
}
interceptor 레지스터:
@Configuration
@EnableWebMvc
public class AppConfig extends WebMvcConfigurerAdapter {
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new ApiLogger()).addPathPatterns("/api/v1/*");
}
}
컨트롤러가 비동기일 경우 Aspect를 사용하여 정확하고 완전한 실행 시간을 가져옵니다.
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import javax.servlet.http.HttpServletRequest;
import java.util.Arrays;
import java.util.UUID;
@Aspect
@Component
public class LoggingAspect {
static Logger log = LoggerFactory.getLogger(LoggingAspect.class);
@Around("execution(* com.aakchoo.api.controller..*(..))")
public Object profileExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable {
long start = System.currentTimeMillis();
String className = joinPoint.getSignature().getDeclaringTypeName();
String methodName = joinPoint.getSignature().getName();
String apiName = className + "."+ methodName;
HttpServletRequest request =
((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();
String requestId = UUID.randomUUID().toString();
log.info("----->>>>>\nREQUESTED_ID: {}\nHOST: {} HttpMethod: {}\nURI: {}\nAPI: {}\nArguments: {}\n",
requestId,
request.getHeader("host"),
request.getMethod(),
request.getRequestURI(),
apiName,
Arrays.toString(joinPoint.getArgs()));
Object result = joinPoint.proceed();
long elapsedTime = System.currentTimeMillis() - start;
log.info("<<<<<-----\nExecution Time: {} ms [REQUESTED_ID: {}] [API: {}]", elapsedTime,requestId,apiName);
return result;
}
}
@EnableAspect 추가JAutoProxy를 응용 프로그램 클래스에 추가
@EnableAsync
@SpringBootApplication
@EnableAspectJAutoProxy
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
build.gradle에는 다음 사항이 필요합니다.
compile 'org.aspectj:aspectjweaver:1.8.10'
compile 'org.springframework.boot:spring-boot-starter-aop'
David의 답변은 정확합니다.필터는 이러한 기능을 Spring Boot에 구현하는 좋은 방법입니다.
Spring Boot에는 다음과 같이 최근 100개의 요구에 대한 정보를 반환하는 엔드포인트가 내장되어 있습니다.
[{
"timestamp": 1394343677415,
"info": {
"method": "GET",
"path": "/trace",
"headers": {
"request": {
"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8",
"Connection": "keep-alive",
"Accept-Encoding": "gzip, deflate",
"User-Agent": "Mozilla/5.0 Gecko/Firefox",
"Accept-Language": "en-US,en;q=0.5",
"Cookie": "_ga=GA1.1.827067509.1390890128; ..."
"Authorization": "Basic ...",
"Host": "localhost:8080"
},
"response": {
"Strict-Transport-Security": "max-age=31536000 ; includeSubDomains",
"X-Application-Context": "application:8080",
"Content-Type": "application/json;charset=UTF-8",
"status": "200"
}
}
}
},{
"timestamp": 1394343684465,
...
}]
애플리케이션이 다른 서비스를 호출하거나 데이터베이스를 조회하는 경우 프로파일링을 위해 Snify를 사용하는 것을 고려할 수 있습니다.서버뿐만 아니라 다운스트림 시스템을 호출하는 데 걸린 시간도 표시됩니다.라이브 데모를 보려면 여기를 클릭하십시오(맨 아래 구석에 있는 검은색 위젯 확인).
면책사항:나는 스니피의 작가이다.
Springboot를 활용할 수 있습니다.InMemoryHttpTraceRepository
우선함으로써add()
방법
public class MyCustomTraceRepository extends InMemoryHttpTraceRepository {
@Override
public void add(HttpTrace trace) {
log(trace);
super.add(trace);
}
//Log only you interested in
public void log(HttpTrace trace) {
Map<String, Object> traceMap = new LinkedHashMap<>();
traceMap.put("status", trace.getResponse().getStatus());
traceMap.put("method", trace.getRequest().getMethod());
traceMap.put("uri", trace.getRequest().getUri().getPath());
traceMap.put("timeTaken", trace.getTimeTaken());
logger.info(traceMap)
}
}
//will Result into
{status=<>, method=<>, uri=/<>, timeTaken=<>}
@요청 및 응답 트레이스 옵션 더 보기:https://docs.spring.io/spring-boot/docs/current/actuator-api/html/ #syslog-syslog
언급URL : https://stackoverflow.com/questions/42857658/how-to-log-time-taken-by-rest-web-service-in-spring-boot
'programing' 카테고리의 다른 글
WordPress 커스텀 페이지에서 "TypeError: $ is not a function" 오류를 수정하는 방법 (0) | 2023.03.09 |
---|---|
Ckeditor 값을 Angular의 모델 텍스트에 바인딩JS 및 레일 (0) | 2023.03.09 |
앱 엔진 Python webapp2에서 JSON을 올바르게 출력하는 방법 (0) | 2023.03.09 |
소품으로 전달된 React 요소에 소품을 추가하는 방법은 무엇입니까? (0) | 2023.03.09 |
"안전" TO_NUMBER() (0) | 2023.03.09 |