# Copyright (C) 2018, 2021 Cancer Care Associates
# 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 Dict, List, Optional, Tuple
from . import connect as _connect
from . import credentials as _credentials
Connection = _connect.Connection
Cursor = _connect.Cursor
[docs]def connect(
hostname: str,
port: int = 1433,
database: str = "MOSAIQ",
alias: Optional[str] = None,
username: Optional[str] = None,
password: Optional[str] = None,
) -> Connection:
"""Connect to a Mosaiq SQL server.
The first time running this function on a system will result in a
prompt to login to the Mosaiq SQL server. The provided credentials
will be stored within the operating system's password storage
facilities. Subsequent calls to this function will pull from that
password storage in order to connect.
Can optionally be called as a context manager. This will close the
database connection once leaving the context manager.
Parameters
----------
hostname : str
The IP address or hostname of the SQL server.
port : int, optional
The port at which the SQL server is hosted, by default ``1433``
database : str, optional
The MSSQL database name, by default ``"MOSAIQ"``
alias : Optional[str], optional
A human readable representation of the server, this is the name
of the server presented to the user should there not be
credentials already on the machine, by default ``"hostname:port/database"``
username : Optional[str], optional
Provide a username to login to the database with, by default the
username is either pulled from the system's credential storage,
or an interactive prompt is used.
password : Optional[str], optional
Provide a password to login to the database with, by default the
password is either pulled from the system's credential storage,
or an interactive prompt is used.
Returns
-------
connection : pymedphys.mosaiq.Connection
A database connection. This connection can be passed to
:func:`pymedphys.mosaiq.execute` to be able to run queries.
The method ``close()`` can be called on this object to close the
database connection.
Examples
--------
See :func:`pymedphys.mosaiq.execute` for examples of usage.
"""
if username is None and password is None:
username, password = _credentials.get_username_password_with_prompt_fallback(
hostname=hostname, port=port, database=database, alias=alias
)
if username is None or password is None:
raise ValueError(
"Must either provide both username and password, or neither of them."
)
connection = _connect.connect_with_credentials(
username, password, hostname=hostname, port=port, database=database
)
return connection
[docs]def execute(
connection: Connection, query: str, parameters: Dict = None
) -> List[Tuple[str, ...]]:
"""Execute SQL queries on a Mosaiq database.
Parameters
----------
connection : pymedphys.mosaiq.Connection
A database connection. This can be retrieved by calling
``pymedphys.mosaiq.connect``
query : str
The SQL query to execute. Do not parse Python variables directly
into this string. Instead create this string as if you
are going to call ``format`` on it
(https://docs.python.org/3/library/stdtypes.html#str.format)
and then utilise the parameters optional argument to this
function. That way, the underlying library ``pymssql`` will
sanitise the parameters to protect against malicious inputs.
parameters : Dict, optional
Parameters to be included within the query. These are sanitised
by the underlying ``pymssql`` library before being included
within the query string, by default None
Returns
-------
results : List[Tuple[str, ...]]
The results from the database query organised so that each row
is an item within the returned list.
Examples
--------
Directly calling the connection object and listing all patients that
have the last name of ``"PHANTOM"``.
>>> import pymedphys.mosaiq
>>> connection = pymedphys.mosaiq.connect('msqsql') # doctest: +SKIP
>>> pymedphys.mosaiq.execute(
... connection,
... '''
... SELECT
... Ident.IDA,
... Patient.Last_Name,
... Patient.First_Name
... FROM Ident, Patient
... WHERE
... Patient.Pat_ID1 = Ident.Pat_ID1 AND
... Patient.Last_Name = %(last_name)s
... ''',
... {"last_name": "PHANTOM"},
... ) # doctest: +SKIP
[('654324', 'PHANTOM', 'CATPHAN'),
('944444', 'PHANTOM', 'DELTA4'),
('654321', 'PHANTOM', 'PELVIS'),
('012534', 'PHANTOM', 'QUASAR4D'),
('987654', 'PHANTOM', 'RESPIRATORY'),
('654325', 'PHANTOM', 'RW3')]
Connecting via a context manager and using pandas to present the
results as a table.
>>> import pandas as pd
>>> import pymedphys.mosaiq
>>> with pymedphys.mosaiq.connect('msqsql') as connection: # doctest: +SKIP
... results = pymedphys.mosaiq.execute(
... connection,
... '''
... SELECT
... Ident.IDA,
... Chklist.Due_DtTm
... FROM Chklist, Staff, Ident
... WHERE
... Chklist.Pat_ID1 = Ident.Pat_ID1 AND
... Staff.Staff_ID = Chklist.Rsp_Staff_ID AND
... Staff.Last_Name = %(qcl_location)s AND
... Chklist.Complete = 0
... ''',
... {"qcl_location": "Physics_Check"},
... )
>>> pd.DataFrame( # doctest: +SKIP
... data=results,
... columns=[
... "patient_id",
... "due",
... ],
... )
patient_id due
0 000000 2021-02-01 23:59:59
1 000001 2021-02-01 23:59:59
2 000002 2021-02-01 23:59:59
3 000003 2021-03-08 23:59:59
"""
with connection.cursor() as cursor:
cursor.execute(query=query, parameters=parameters)
results: List[Tuple[str, ...]] = cursor.fetchall()
return results