data_validation/raw_query.py (34 lines of code) (raw):
# Copyright 2024 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from typing import TYPE_CHECKING
from sqlalchemy.engine.row import Row
from data_validation import clients, consts, state_manager
if TYPE_CHECKING:
from argparse import Namespace
def run_raw_query_against_connection(args: "Namespace") -> list:
"""Return results of raw query for ad hoc usage."""
mgr = state_manager.StateManager()
with clients.get_data_client_ctx(mgr.get_connection_config(args.conn)) as client:
cursor = client.raw_sql(args.query)
res = cursor.fetchall()
try:
cursor.close()
except Exception:
pass
return res
def _minimize_raw_query_output(query_output: list) -> list:
"""Reduce single column output from list(tuple(_)) to list(_)"""
if not query_output:
return query_output
if isinstance(query_output[0], (list, tuple)) and len(query_output[0]) == 1:
return [_[0] for _ in query_output]
else:
return query_output
def print_raw_query_output(query_output: list, format: str = consts.FORMAT_TYPE_PYTHON):
"""Print a query resultset avoiding SQLAlchemy "... (nn characters truncated) ..." behaviour.
Args:
query_output (list): A set of rows from a SQLAlchemy query.
format (str): Output format:
FORMAT_TYPE_PYTHON: Legacy format matching previous behaviour, simple print the Python object.
FORMAT_TYPE_MINIMAL: Attempt to minimize additional noise for bare minimum output.
This is useful for a specific customer who are using DVT raw queries to
generate other DVT commands and don't want any escaping or extra parentheses
in the output.
"""
def row_to_tuple(row) -> tuple:
"""This prevents SQLAlchemy string truncation inside Row() objects by first converting them to a tuple."""
if isinstance(row, (Row, list)):
return tuple(row)
else:
return row
sanitized_output = [row_to_tuple(_) for _ in query_output or []]
if format == consts.FORMAT_TYPE_MINIMAL:
for row in _minimize_raw_query_output(sanitized_output):
print(row)
else:
print(sanitized_output)