To spot N+1 select, QuickPerf live verifies if the application emits the same type of select statement with different parameter values several times. The same select type executed in a loop will also result in a positive detection.
The quickperf.database.n+1.detected property allows to enable the detection. quickperf.database.n+1.threshold properties is optional. Its default value is 3. If the number of generated SELECT statements is less than the configured threshold, when no warning will be raised.
🔧 .properties configuration example
quickperf.database.n+1.detected=true
quickperf.database.n+1.threshold=20
🔧 YAML configuration example
quickperf:
database:
n+1:
detected: true
threshold: 10
🔧 MBean configuration
QuickPerf
-- Database
-- Operations
-- boolean isNPlusOneSelectDetected()
-- void setNPlusOneSelectDetected(boolean)
-- int getNPlusOneDetectionThreshold()
-- void setNPlusOneDetectionThreshold(int)
🔎 Log example
2021-10-18 19:27:18.903 WARN 21632 --- [nio-8080-exec-1] s.QuickPerfHttpCallHttpCallWarningLogger :
GET 200 http://localhost:8080/owners?lastName=
* [WARNING] N+1 select suspicion - 453 SELECT]
🔧 .properties configuration
quickperf.database.sql.displayed=true
🔧 YAML configuration
quickperf:
database:
sql:
displayed: true
🔧 MBean configuration
QuickPerf
-- Database
-- Operations
-- boolean isDatabaseConnectionProfiled()
-- void setDatabaseConnectionProfiled(boolean)
🔎 Log example
2021-10-27 14:51:48.595 INFO 6020 --- [nio-8080-exec-7] .w.s.QuickPerfHttpCallHttpCallInfoLogger :
GET 200 http://localhost:8080/owners?lastName=
* DATABASE CONNECTION PROFILING
connection 905860618 - javax.sql.DataSource.getConnection()
com.zaxxer.hikari.HikariDataSource$$EnhancerBySpringCGLIB$$830fe3d5.getConnection(<generated>)
org.hibernate.engine.jdbc.connections.internal.DatasourceConnectionProviderImpl.getConnection(DatasourceConnectionProviderImpl.java:122)
org.hibernate.internal.NonContextualJdbcConnectionAccess.obtainConnection(NonContextualJdbcConnectionAccess.java:38)
org.hibernate.resource.jdbc.internal.LogicalConnectionManagedImpl.acquireConnectionIfNeeded(LogicalConnectionManagedImpl.java:108)
connection 905860618 - java.sql.Connection.setReadOnly(boolean readOnly) [readOnly: true]
org.springframework.jdbc.datasource.DataSourceUtils.prepareConnectionForTransaction(DataSourceUtils.java:188)
org.springframework.orm.jpa.vendor.HibernateJpaDialect.beginTransaction(HibernateJpaDialect.java:153)
org.springframework.orm.jpa.JpaTransactionManager.doBegin(JpaTransactionManager.java:421)
org.springframework.transaction.support.AbstractPlatformTransactionManager.startTransaction(AbstractPlatformTransactionManager.java:400)
connection 905860618 - java.sql.Connection.setAutoCommit(boolean autoCommit) [autoCommit: false]
org.hibernate.resource.jdbc.internal.AbstractLogicalConnectionImplementor.begin(AbstractLogicalConnectionImplementor.java:72)
org.hibernate.resource.jdbc.internal.LogicalConnectionManagedImpl.begin(LogicalConnectionManagedImpl.java:283)
org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl$TransactionDriverControlImpl.begin(JdbcResourceLocalTransactionCoordinatorImpl.java:246)
org.hibernate.engine.transaction.internal.TransactionImpl.begin(TransactionImpl.java:83)
connection 905860618 - java.sql.Connection.commit() [isolation: transaction_read_committed]
org.hibernate.resource.jdbc.internal.AbstractLogicalConnectionImplementor.commit(AbstractLogicalConnectionImplementor.java:86)
org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl$TransactionDriverControlImpl.commit(JdbcResourceLocalTransactionCoordinatorImpl.java:282)
org.hibernate.engine.transaction.internal.TransactionImpl.commit(TransactionImpl.java:101)
org.springframework.orm.jpa.JpaTransactionManager.doCommit(JpaTransactionManager.java:562)
connection 905860618 - java.sql.Connection.setAutoCommit(boolean autoCommit) [autoCommit: true]
org.hibernate.resource.jdbc.internal.AbstractLogicalConnectionImplementor.resetConnection(AbstractLogicalConnectionImplementor.java:106)
org.hibernate.resource.jdbc.internal.LogicalConnectionManagedImpl.afterCompletion(LogicalConnectionManagedImpl.java:288)
org.hibernate.resource.jdbc.internal.AbstractLogicalConnectionImplementor.commit(AbstractLogicalConnectionImplementor.java:95)
org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl$TransactionDriverControlImpl.commit(JdbcResourceLocalTransactionCoordinatorImpl.java:282)
connection 905860618 - java.sql.Connection.setReadOnly(boolean readOnly) [readOnly: false]
org.springframework.jdbc.datasource.DataSourceUtils.resetConnectionAfterTransaction(DataSourceUtils.java:252)
org.springframework.orm.jpa.vendor.HibernateJpaDialect$SessionTransactionData.resetSessionState(HibernateJpaDialect.java:371)
org.springframework.orm.jpa.vendor.HibernateJpaDialect.cleanupTransaction(HibernateJpaDialect.java:214)
org.springframework.orm.jpa.JpaTransactionManager.doCleanupAfterCompletion(JpaTransactionManager.java:642)
connection 905860618 - java.sql.Connection.close()
org.hibernate.engine.jdbc.connections.internal.DatasourceConnectionProviderImpl.closeConnection(DatasourceConnectionProviderImpl.java:127)
org.hibernate.internal.NonContextualJdbcConnectionAccess.releaseConnection(NonContextualJdbcConnectionAccess.java:49)
org.hibernate.resource.jdbc.internal.LogicalConnectionManagedImpl.releaseConnection(LogicalConnectionManagedImpl.java:217)
org.hibernate.resource.jdbc.internal.LogicalConnectionManagedImpl.close(LogicalConnectionManagedImpl.java:259)
🔧 .properties configuration
quickperf.database.sql.displayed=true
🔧 YAML configuration
quickperf:
database:
sql:
displayed: true
🔧 MBean configuration
QuickPerf
-- Database
-- Operations
-- boolean isSqlDisplayed()
-- void setSqlDisplayed(boolean)
🔎 Log example
2021-10-14 20:35:07.849 INFO 5068 --- [nio-8080-exec-3] .w.s.QuickPerfHttpCallHttpCallInfoLogger :
GET 200 http://localhost:8080/owners?lastName=
* SQL
Time:25, Success:True, Type:Prepared, Batch:False, QuerySize:1, BatchSize:0, Query:["
select
distinct owner0_.id as id1_0_0_,
pets1_.id as id1_1_1_,
owner0_.first_name as first_na2_0_0_,
owner0_.last_name as last_nam3_0_0_,
owner0_.address as address4_0_0_,
owner0_.city as city5_0_0_,
owner0_.telephone as telephon6_0_0_,
pets1_.name as name2_1_1_,
pets1_.birth_date as birth_da3_1_1_,
pets1_.owner_id as owner_id4_1_1_,
pets1_.type_id as type_id5_1_1_,
pets1_.owner_id as owner_id4_1_0__,
pets1_.id as id1_1_0__
from
owners owner0_
left outer join
pets pets1_
on owner0_.id=pets1_.owner_id
where
owner0_.last_name like ?"], Params:[(%)]
Time:9, Success:True, Type:Prepared, Batch:False, QuerySize:1, BatchSize:0, Query:["
select
pettype0_.id as id1_3_0_,
pettype0_.name as name2_3_0_
from
types pettype0_
where
pettype0_.id=?"], Params:[(1)]
🔧 .properties configuration
quickperf.database.sql.displayed.selected-columns=true
🔧 YAML configuration
quickperf:
database:
sql:
displayed.selected-columns: true
🔧 MBean configuration
QuickPerf
-- Database
-- Operations
-- void setSelectedColumnsDisplayed(boolean)
-- boolean setSelectedColumnsDisplayed()
🔎 Log example
2021-10-14 20:19:57.504 INFO 5068 --- [nio-8080-exec-5] .w.s.QuickPerfHttpCallHttpCallInfoLogger :
GET 200 http://localhost:8080/owners?lastName=
* SELECTED COLUMNS
╔════════╤══════════════════════════════════════════════════════════╤════╗
║ Table │ Selected columns │ Nb ║
╠════════╪══════════════════════════════════════════════════════════╪════╣
║ TYPES │ ID * NAME │ 2 ║
╟────────┼──────────────────────────────────────────────────────────┼────╢
║ PETS │ TYPE_ID * BIRTH_DATE * ID * NAME * OWNER_ID │ 5 ║
╟────────┼──────────────────────────────────────────────────────────┼────╢
║ OWNERS │ TELEPHONE * CITY * ADDRESS * ID * LAST_NAME * FIRST_NAME │ 6 ║
╚════════╧══════════════════════════════════════════════════════════╧════╝
🔧 .properties configuration example
You have to configure two properties to spot long SQL queries.
🔧 .properties configuration
quickperf.database.sql.execution-time.detected
quickperf.database.sql.execution-time.detected=true
#ping value included in execution time threshold
quickperf.database.sql.execution-time.thresholdInMs=50
🔧 YAML configuration example
quickperf:
execution-time:
detected: true
thresholdInMs: 50
🔧 MBean configuration
QuickPerf
-- Database
-- Operations
-- boolean isSqlDisplayed()
-- void setSqlExecutionTimeThresholdInMilliseconds(int)
🔎 Log example
2021-10-18 19:27:18.903 WARN 21632 --- [nio-8080-exec-1] s.QuickPerfHttpCallHttpCallWarningLogger :
GET 200 http://localhost:8080/owners?lastName=
* [WARNING] At least one SQL query has an execution time greater than 50 ms
Time:158, Success:True, Type:Prepared, Batch:False, QuerySize:1, BatchSize:0, Query:["
select
distinct owner0_.id as id1_0_0_,
pets1_.id as id1_1_1_,
owner0_.first_name as first_na2_0_0_,
owner0_.last_name as last_nam3_0_0_,
owner0_.address as address4_0_0_,
owner0_.city as city5_0_0_,
owner0_.telephone as telephon6_0_0_,
pets1_.name as name2_1_1_,
pets1_.birth_date as birth_da3_1_1_,
pets1_.owner_id as owner_id4_1_1_,
pets1_.type_id as type_id5_1_1_,
pets1_.owner_id as owner_id4_1_0__,
pets1_.id as id1_1_0__
from
owners owner0_
left outer join
pets pets1_
on owner0_.id=pets1_.owner_id
where
owner0_.last_name like ?"], Params:[(%)]
🔧 .properties configuration example
You have to configure two properties to set a threshold to SQL executions. The default value for this threshold is 10 SQL executions.
🔧 .properties configuration
quickperf.database.sql.execution.detected=true
quickperf.database.sql.execution.threshold=10
🔧 YAML configuration example
quickperf:
database:
sql:
execution:
detected: true
threshold: 10
🔧 MBean configuration
QuickPerf
-- Database
-- Operations
-- boolean isSqlExecutionDetected()
-- void setSqlExecutionDetected(boolean)
-- int getSqlExecutionThreshold()
-- void setSqlExecutionThreshold(int)
🔎 Log example
2022-10-19 15:09:16.100 INFO 68225 --- [ restartedMain] o.q.s.s.ASpringBootApplication : Started ASpringBootApplication in 7.06 seconds (JVM running for 9.127)
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select player0_.id as id1_0_, player0_.first_name as first_na2_0_, player0_.last_name as last_nam3_0_, player0_.team_id as team_id4_0_ from player player0_
Hibernate: select team0_.id as id1_1_0_, team0_.name as name2_1_0_ from team team0_ where team0_.id=?
2022-10-19 15:10:38.899 WARN 68225 --- [tp1496937262-36] s.QuickPerfHttpCallHttpCallWarningLogger :
GET 200 http://localhost:8080/json-with-n-plus-one-select
* [WARNING] You have reached your SQL executions threshold : 11 > 10
The use of bind parameters in queries is generally a good practice for security and performance reasons. QuickPerf live can detect queries without bind parameters with the following property:
🔧 .properties configuration example
quickperf.database.sql.without-bind-param.detected=true
🔧 YAML configuration example
quickperf:
database:
sql:
without-bind-param:
detected: true
🔧 MBean configuration
QuickPerf
-- Database
-- Operations
-- boolean isSqlWithoutBindParamDetected()
-- void setSqlWithoutBindParamDetected(boolean)
🔎 Log example
Hibernate: SELECT * FROM Player p where p.lastName like '%ronaldo%'
2022-12-14 13:59:19.776 WARN 13632 --- [qtp640585435-24] s.QuickPerfHttpCallHttpCallWarningLogger :
GET 200 http://localhost:8080/player/unbounded-params?name=ronaldo
* [WARNING] Some of your SQL queries don't use bind parameters.