Improve the usage, output, comparison, and logic of assertQueries.
Review Request #13368 — Created Oct. 23, 2023 and submitted — Latest diff uploaded
assertQueries()has been a useful tool for auditing database queries
and making sure they don't regress, but there have been a few issues in
using it:
In the
whereclause, empty or deeply-nested query clauses (such as
Q(Q(Q(a=1)))could be annoying to compare.
If the number of executed queries and the number of expected queries
did not match, the tool would assert early and fail to show results.
Output was shown with the expected queryset field value on the
left-hand side of a comparison, and the executed on the right-hand
side, which is the inverse of how assertions usually show results.
Output was not sorted, which could make it difficult to keep or
compare to sorted queries.
It wasn't easy to determine where a query actually came from.
There were no type hints, and the code was difficult to maintain.
This change addresses all of that.
Query where clauses are now normalized for easier comparison, with empty
clauses and deep nesting cleaned up. They're now shown over multiple
lines, one clause per lineMismatches in query counts are reported but do not prevent comparison.
This makes it easy to just run with an empty list and see all the
queries made.Output of queryset comparisons is now in a standard order, and are
sorted.New
with_tracebacksandtracebacks_sizearguments allow the caller
to output traceback information to help track down the origin of a
query.Type hints are added. These are sort of loose, meant mostly for
documentation purposes. The reason being that, in order to support
both pyright and dynamically-constructed queries, the function has to
accept general dictionary, as pyright won't infer the right-hand side
from the left when concatenating lists.Until/unless this is resolved, we type hint with the ideal type for
these entries, but accept dictionaries. This means that invalid keys or
values won't be caught, unfortunately, but it does give us
documentaiton.Internally, much of the code has been cleaned up. A utility function has
been added to perform the comparisons in a type-safe manner and to
record results. It's now easier to reason about this logic and to keep
it updated.Overall, this change does a lot to improve the usability of query
assertions.
All Djblets and Review Board unit tests pass.
Used this for all the new in-progress query performance optimization
tests for Review Board 5.0.7 / 6.0.1.
