Skip to content

Latest commit

 

History

History
394 lines (332 loc) · 15.1 KB

database_diagnostics.md

File metadata and controls

394 lines (332 loc) · 15.1 KB

Database diagnostics

N+1 select detection

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]

Database connection profiling

🔧 .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)

SQL query

Display

🔧 .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)]

Selected columns

🔧 .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  ║
  ╚════════╧══════════════════════════════════════════════════════════╧════╝

Execution time

🔧 .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:[(%)]

Execution Threshold

🔧 .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

Missing bind parameters in queries

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.