Skip to main content

Lightweight database query profiler.

Project description

Python 3.7+ Poetry application-tests coverage GitHub last commit

code style: prettier code style: black Imports: isort pre-commit.ci status Sourcery


Database Query Profiler 🗃️⏱️

Lightweight database query profiler.

This tool is database-agnostic -- just provide a class that connects to your database with an execute method, and the queries that you want to profile.

This is NOT a replacement for analysing the query plan. This should just support the analysis done with it.

Installation ⬇️

This is currently only available on GitHub, so you'll need to supply the GitHub URL to pip:

pip install git+https://github.com/Bilbottom/db-query-profiler@v0.0.4

Sample Output 📝

Given a set of queries (details below), this package prints the average time in seconds taken to run each query, as well as the percentage of the total time taken by each query.

The tqdm package is used to show progress of the queries being run.

A typical output will look something like this:

Start time: 2023-05-07 12:38:06.879738
----------------------------------------
100%|██████████| 5/5 [00:01<00:00,  3.29it/s]
query-1.sql: 0.10063192s (33.4%)
query-2.sql: 0.20044784s (66.6%)
----------------------------------------
End time: 2023-05-07 12:38:08.757555

Usage 📖

The package exposes a single function, time_queries, which currently requires:

  1. A database connection/cursor class that implements an execute method.
  2. The number of times to re-run each query.
  3. A directory containing the SQL files with the queries to run.

There should only be a single query in each file, and the file name will be used as the query name in the output.

For the following examples, assume that there are SQL files in the queries directory.

SQLite Example

Official documentation: https://docs.python.org/3/library/sqlite3.html

import sqlite3

import db_query_profiler


def main() -> None:
    db_conn = sqlite3.connect(":memory:")  # Or a path to a database file
    db_query_profiler.time_queries(
        conn=db_conn,
        repeat=5,
        directory="queries"
    )


if __name__ == "__main__":
    main()

Snowflake Example

Official documentation: https://docs.snowflake.com/en/developer-guide/python-connector/python-connector-example

Some databases, like Snowflake, have extra layers of caching that can affect the results of the profiling. To avoid this and make the runtime comparisons more genuine, it's recommended to turn off these extra caching options (where this is supported).

import db_query_profiler
import snowflake.connector  # snowflake-connector-python


# This dictionary is just for illustration purposes and
# you should use whatever connection method you prefer
CREDENTIALS = {
    "user": "XXX",
    "password": "XXX",
    "account": "XXX",
    "warehouse": "XXX",
    "role": "XXX",
    "database": "XXX",
}


def main() -> None:
    db_conn = snowflake.connector.SnowflakeConnection(**CREDENTIALS)
    with db_conn.cursor() as cursor:
        cursor.execute("""ALTER SESSION SET USE_CACHED_RESULT = FALSE;""")
        db_query_profiler.time_queries(
            conn=cursor,
            repeat=5,
            directory="queries",
        )
        cursor.execute("""ALTER SESSION SET USE_CACHED_RESULT = TRUE;""")
    db_conn.close()


if __name__ == "__main__":
    main()

Warnings ⚠️

This package will open and run all the files in the specified directory, so be careful about what you put in there -- potentially unsafe SQL commands could be run.

This package only reads from the database, so it's encouraged to configure your database connection in a read-only way.

SQLite

Official documentation:

To connect to a SQLite database in a read-only way, use the uri=True parameter with file: and ?mode=ro surrounding the database path when connecting:

db_conn = sqlite3.connect("file:path/to/database.db?mode=ro", uri=True)

Contributing 🤝

The Python packaging is managed with Poetry (check which version in the poetry.lock file), but that should be the only dependency.

To get started, just clone the repo, install the dependencies, and enable pre-commit:

poetry install --sync --with dev,test
pre-commit install --install-hooks

Happy coding! 🎉

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

db_query_profiler-0.0.4.tar.gz (6.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

db_query_profiler-0.0.4-py3-none-any.whl (7.3 kB view details)

Uploaded Python 3

File details

Details for the file db_query_profiler-0.0.4.tar.gz.

File metadata

  • Download URL: db_query_profiler-0.0.4.tar.gz
  • Upload date:
  • Size: 6.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.18

File hashes

Hashes for db_query_profiler-0.0.4.tar.gz
Algorithm Hash digest
SHA256 126b953dcc65363b6a1e8f4500f73714eed840df9d473f275496eeae3d1eb257
MD5 f3bc83f4812d627d0fa7cbe643e93d38
BLAKE2b-256 c7eb691f5fb97ffa74e58c4f3e55eae94f1e423e1609d97bfec541395bf12785

See more details on using hashes here.

File details

Details for the file db_query_profiler-0.0.4-py3-none-any.whl.

File metadata

File hashes

Hashes for db_query_profiler-0.0.4-py3-none-any.whl
Algorithm Hash digest
SHA256 5982afcf7f53ca5facf32efd8a1413dc75cab5d4bbe6d20a3a817c0adb6672d4
MD5 336da0708b74fd4c2bdb5014b17d9701
BLAKE2b-256 c9513eaad27d9aaac64c3ddc12a22e422981b7a9d5d95552a1195dadacf5f068

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page