API 개발 후 jMeter를 이용하여 성능테스트를 하던 중 아래와 같은 오류 메시지가 발생했습니다.
hikari-pool-1 - Connection is not available, request timed out after 30000ms.
장애의 원인
장애의 원인을 분석하던 중 아래 3가지 항목 간 상관관계가 있다는 것을 알았습니다.
- HikariCP maximum pool size
- DB에 insert하고자 하는 전체 Thread Count
- 하나의 Task에서 동시에 필요한 Connection 수
부하 상황에서 Thread간 Connection을 차지 하기 위한 Race Condition(경쟁 상태)가 발생합니다.
Thread가 DB를 사용해야 할 때 사용 가능한 Connection이 없으면 사용가능한 Connection이 발생할 때까지 기다리다가 설정된 timeout 시간에 도달하면 Exception을 발생시킵니다.
결론부터 말하면, Thread의 갯수보다 HikariCP의 maximum pool size가 적은 경우 장애가 발생하게 됩니다.
위의 1,2,3번에 값을 대입하면 아래와 같습니다.
- HikariCP Maximum pool size: 1개
- Thread Count: 1개
- 하나의 Task에서 동시에 요구되는 Connection 수: 2개
이 경우 필요한 Connection은 2개인데, 실제 connection은 1개밖에 없기 때문에 장애가 발생하는 것입니다.
JPA를 사용하는 경우 예상보다 많은 Connection이 발생할 수 있는데 그 예를 들면 아래와 같습니다.
@Entity
class Message {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private long id;
private String title;
private String contents;
}
@Transactional
public Message save(Message msg) {
return repository.save(msg);
}
위와 같은 코드가 있을 떄 repository.save를 하는데 몇개의 Connection이 필요할까요?
정답은 2개 입니다.
코드만 보면 하나의 Connection으로 Insert가 될 것 같은데요
@GeneratedValue(strategy = GenerationType.AUTO)
바로 이 코드 때문에 Connection이 하나 더 발생하게 됩니다.
JPA에서 DB Insert 시, id 생성 방식을 결정하는 Annotation입니다.
GenerationType이 AUTO이고, id변수의 Type이 Long이기 때문에 내부적으로는 Sequence를 기반으로 ID를 생성합니다. 하지만 MySQL 등 Sequence를 지원하지 않는 DB의 경우 테이블전략(default: hibernate_sequence)을 이용하여 ID값을 생성합니다.
[추가] GenerationType.AUTO에 대하여
Springboot 2.x.x 부터 MSSQL에서의 GenerationType.AUTO는 IDENTITY가 아닌 TABLE을 기본 시퀀스 전략으로 사용합니다.
그 이론은 다음과 같습니다.
- Springboot 1.5.x : hibernate.id.new_generator_mappings = false
- Springboot 2.x.x: hibernate.id.new_generator_mappings = true
hibernate.id.new_generator_mappings = false일 경우
이 경우 사용하는 DB(Dialect)에 의해 Generator가 결정됩니다.
Dialect에서 supportsIdentityColumns()가 true 인 경우 IdentityGenerator를 사용하게 됩니다.
false 인 경우 SequenceStyleGenerator를 사용하게 됩니다.
hibernate.id.new_generator_mappings = true일 경우
이 경우 Sequence 기능을 지원하는 경우 SequenceGenerator를 사용하고,
Sequence 기능을 사용하지 않는 경우 TableGenerator를 사용합니다.
여기서 hibernate_sequence 테이블을 조회 및 insert/upate를 하면서 Sub Transaction을 생성하여 실행되게 됩니다.
select next_val as id_val from hibernate_sequence for update;
MySQL for update 쿼리는 조회한 row에 lock을 걸어 현재 트랜잭션이 끝나기 전까지 다른 session의 접근을 막습니다.(동시에 여러 Thread가 ID를 구할 때, 순차적인 ID를 구하기 위함)
이미 이 이슈는 HikariCP github에서도 issue로 등록되었고, HikariCP wiki에서 Dead lock을 해결하는 방법을 제시하고 있습니다.
- Github: https://github.com/brettwooldridge/HikariCP
- issue: https://github.com/brettwooldridge/HikariCP/issues/442#issuecomment-146096704
- wiki: https://github.com/brettwooldridge/HikariCP/wiki/About-Pool-Sizing
wiki에서 제시 된 공식은 아래와 같습니다.
pool size = Tn x (Cm - 1) + 1
- Tn : 전체 Thread 갯수
- Cm : 하나의 Task에서 동시에 필요한 Connection 수
HikiriCP wiki에서는 이 공식에서 나온 수 이상의 값을 Maximum pool size로 설정하면 Dead lock을 피할 수 있다고 합니다.
말 그대로 이 값은 최소값입니다. 너무 많은 pool size도 좋지 않지만, 여유있게 pool size를 잡아주는 것이 좋을 것 같습니다.
Spring boot에서 설정 방법
application.properties에 아래 내용을 추가합니다.
server.tomcat.threads.max=30
spring.datasource.hikari.maximum-pool-size=100
#저의 경우에는 아래 코드는 동작하지 않았습니다.
#spring.datasource.tomcat.max-active=100
이렇게 설정한 경우 Spring boot에서 Run As > Spring Boot App 를 실행했을 때는 정상동작 하지만, WAR로 만들어 Tomcat에 배포 했을 때 정상동작 하지 않았습니다.
확인 해보니, spring.datasource.hikari.maximum-pool-size는 정상적으로 설정이 되었지만, server.tomcat.threads.max가 설정이 되지 않았습니다.
따라서, tomcat설치경로/conf/server.xml에 아래와 같이 maxThreads를 설정했습니다.
<Connector port="8092" protocol="HTTP/1.1"
connectionTimeout="30000"
maxThreads="30"
redirectPort="8443" />
참고
HikariCP Dead lock에서 벗어나기 (이론편)
https://woowabros.github.io/experience/2020/02/06/hikaricp-avoid-dead-lock.html
HikariCP Dead lock에서 벗어나기 (실전편)
https://woowabros.github.io/experience/2020/02/06/hikaricp-avoid-dead-lock-2.html
Spring-Boot: How do I set JDBC pool properties like maximum number of connections?
apache-tomcat 공식 document
https://tomcat.apache.org/tomcat-8.5-doc/config/http.html
Spring Boot Data JPA 2.0 에서 id Auto_increment 문제 해결
https://jojoldu.tistory.com/295
'Database ( DB ) > Database' 카테고리의 다른 글
mssql] Adding an identity to an existing column (0) | 2021.01.01 |
---|---|
springboot에 flyway 사용 (0) | 2020.11.02 |
HikariCP 소개 (0) | 2020.10.17 |
Spring boot Data JPA 2.0에서 auto_increment 문제 해결 (2) | 2020.10.14 |
SQL Server JDBC Driver(MSSQL)에서의 NVARCHAR, VARCHAR (0) | 2020.10.10 |