Skip to content

Latest commit

 

History

History
435 lines (318 loc) · 15.7 KB

tracing.rst

File metadata and controls

435 lines (318 loc) · 15.7 KB

Tracing python-oracledb

Application Tracing

There are multiple approaches for application tracing and monitoring:

Oracle Database End-to-End Tracing

Oracle Database end-to-end application tracing simplifies diagnosing application code flow and performance problems in multi-tier or multi-user environments.

The connection attributes :attr:`Connection.client_identifier`, :attr:`Connection.clientinfo`, :attr:`Connection.dbop`, :attr:`Connection.module`, and :attr:`Connection.action` set metadata for end-to-end tracing. The values can be queried from data dictionary and dynamic performance views to monitor applications, or you can use tracing utilities. Values may appear in logs and audit trails.

Also see :ref:`appcontext` for information about setting Application Contexts.

The :attr:`~Connection.client_identifier` attribute is typically set to the name (or identifier) of the actual end user initiating a query. This allows the database to distinguish, and trace, end users for applications that connect to a common database username. It can also be used by Oracle Virtual Private Database (VPD) policies to automatically limit data access.

The :attr:`~Connection.module` and :attr:`~Connection.action` attributes can be set to user-chosen, descriptive values identifying your code architecture.

After attributes are set, the values are sent to the database when the next :ref:`round-trip <roundtrips>` to the database occurs, for example when the next SQL statement is executed.

The attribute values will remain set in connections released back to a connection pool. When the application re-acquires a connection from the pool, it should initialize the values to a desired state before using that connection.

The example below shows setting the action, module, and client identifier attributes on a connection object, and then querying a view to see the recorded values. The example both sets and queries the values, but typically monitoring is done externally to the application.

# Set the tracing metadata
connection.client_identifier = "pythonuser"
connection.action = "Query Session tracing parameters"
connection.module = "End-to-end Demo"

for row in cursor.execute("""
        SELECT username, client_identifier, module, action
        FROM V$SESSION
        WHERE SID = SYS_CONTEXT('USERENV', 'SID')"""):
    print(row)

The output will be like:

('SYSTEM', 'pythonuser', 'End-to-end Demo', 'Query Session tracing parameters')

The values can also be manually set by calling DBMS_APPLICATION_INFO procedures or DBMS_SESSION.SET_IDENTIFIER. These incur round-trips to the database which reduces application scalability:

BEGIN
    DBMS_SESSION.SET_IDENTIFIER('pythonuser');
    DBMS_APPLICATION_INFO.set_module('End-to-End Demo');
    DBMS_APPLICATION_INFO.set_action(action_name => 'Query Session tracing parameters');
END;

The :attr:`Connection.dbop` attribute can be used for Real-Time SQL Monitoring, see Monitoring Database Operations. The value will be shown in the DBOP_NAME column of the V$SQL_MONITOR table:

connection.dbop = "my op"

for row in cursor.execute("""
        SELECT dbop_name
        FROM v$sql_monitor
        WHERE SID = SYS_CONTEXT('USERENV', 'SID')"""):
    print(row)

Subclassing Connections

Subclassing enables applications to add "hooks" for connection and statement execution. This can be used to alter or log connection and execution parameters, and to extend python-oracledb functionality.

The example below demonstrates subclassing a connection to log SQL execution to a file. This example also shows how connection credentials can be embedded in the custom subclass, so application code does not need to supply them.

class Connection(oracledb.Connection):
    log_file_name = "log.txt"

    def __init__(self):
        connect_string = "hr/hr_password@dbhost.example.com/orclpdb"
        self._log("Connect to the database")
        return super(Connection, self).__init__(connect_string)

    def _log(self, message):
        with open(self.log_file_name, "a") as f:
            print(message, file=f)

    def execute(self, sql, parameters):
        self._log(sql)
        cursor = self.cursor()
        try:
            return cursor.execute(sql, parameters)
        except oracledb.Error as e:
            error_obj, = e.args
            self._log(error_obj.message)
            raise

connection = Connection()
connection.execute("""
        select department_name
        from departments
        where department_id = :id""", dict(id=270))

The messages logged in log.txt are:

Connect to the database

            select department_name
            from departments
            where department_id = :id

If an error occurs, perhaps due to a missing table, the log file would contain instead:

Connect to the database

            select department_name
            from departments
            where department_id = :id
ORA-00942: table or view does not exist

In production applications, be careful not to log sensitive information.

See Subclassing.py for an example.

Debugging PL/SQL with the Java Debug Wire Protocol

The Java Debug Wire Protocol (JDWP) for debugging PL/SQL can be used with python-oracledb.

Python-oracledb applications that call PL/SQL can step through that PL/SQL code using JDWP in a debugger. This allows Python and PL/SQL code to be debugged in the same debugger environment. You can enable PL/SQL debugging in python-oracledb as follows:

  • In python-oracledb Thin or Thick modes, set the ORA_DEBUG_JDWP environment variable to host=hostname;port=portnum indicating where the PL/SQL debugger is running. Then run the application.
  • In python-oracledb Thin mode, you can alternatively set the connection parameter debug_jdwp during connection. This variable defaults to the value of the ORA_DEBUG_JDWP environment variable.

See the documentation on DBMS_DEBUG_JDWP, the video PL/SQL debugging with Visual Studio and Visual Studio Code, and the blog post Debugging PL/SQL with Visual Studio Code (and more).

Low Level SQL Tracing

The Thick mode of python-oracledb is implemented using the ODPI-C wrapper on top of the Oracle Client libraries. The ODPI-C tracing capability can be used to log executed python-oracledb statements to the standard error stream. Before executing Python, set the environment variable DPI_DEBUG_LEVEL to 16.

At a Windows command prompt, this could be done with:

set DPI_DEBUG_LEVEL=16

On Linux, you might use:

export DPI_DEBUG_LEVEL=16

After setting the variable, run the Python Script, for example on Linux:

python end-to-endtracing.py 2> log.txt

For an application that does a single query, the log file might contain a tracing line consisting of the prefix 'ODPI', a thread identifier, a timestamp, and the SQL statement executed:

ODPI [26188] 2019-03-26 09:09:03.909: ODPI-C 3.1.1
ODPI [26188] 2019-03-26 09:09:03.909: debugging messages initialized at level 16
ODPI [26188] 2019-03-26 09:09:09.917: SQL SELECT * FROM jobss
Traceback (most recent call last):
File "end-to-endtracing.py", line 14, in <module>
  cursor.execute("select * from jobss")
oracledb.DatabaseError: ORA-00942: table or view does not exist

See ODPI-C Debugging for documentation on DPI_DEBUG_LEVEL.

Using Connection Identifiers

A unique connection identifier (CONNECTION_ID) is generated for each connection to the Oracle Database. The connection identifier is shown in some Oracle Network error messages and logs, which helps in better tracing and diagnosing of connection failures. For example:

DPY-6005: cannot connect to database (CONNECTION_ID=m0PfUY6hYSmWPcgrHZCQIQ==)

You can define a prefix value which is added to the beginning of the CONNECTION_ID. This prefix aids in identifying the connections from a specific application.

In python-oracledb Thin mode, you can specify a prefix in the connection_id_prefix parameter when creating :meth:`standalone connections <oracledb.connect()>`, or :meth:`pooled connections <oracledb.create_pool()>`. Also, you can specify the connection identifier in :meth:`oracledb.ConnectParams()` or :meth:`oracledb.PoolParams()`. For example:

connection = oracledb.connect(user="hr", password=userpwd,
                              dsn="localhost/orclpdb",
                              connection_id_prefix="MYAPP")

If this connection to the database fails, MYAPP is added as a prefix to the CONNECTION_ID as shown in the error message below:

DPY-6005: cannot connect to database (CONNECTION_ID=MYAPPm0PfUY6hYSmWPcgrHZCQIQ==).

In python-oracledb Thick mode, you can specify the connection identifier prefix in a connection string. For example:

mydb = (DESCRIPTION =
         (ADDRESS_LIST= (ADDRESS=...) (ADDRESS=...))
         (CONNECT_DATA=
            (SERVICE_NAME=sales.us.example.com)
            (CONNECTION_ID_PREFIX=MYAPP)
         )
       )

Depending on the Oracle Database version in use, the information that is shown in logs varies.

See Troubleshooting Oracle Net Services for more information on connection identifiers.

Finding the python-oracledb Mode

The boolean attributes :attr:`Connection.thin` and :attr:`ConnectionPool.thin` can be used to show the current mode of a python-oracledb connection or pool, respectively. The python-oracledb version can be shown with :data:`oracledb.__version__`.

The information can also be seen in the Oracle Database data dictionary table V$SESSION_CONNECT_INFO:

with connection.cursor() as cursor:
    sql = """SELECT UNIQUE CLIENT_DRIVER
             FROM V$SESSION_CONNECT_INFO
             WHERE SID = SYS_CONTEXT('USERENV', 'SID')"""
    for r, in cursor.execute(sql):
        print(r)

In the python-oracledb Thin mode, the output will be:

python-oracledb thn : 1.0.0

In the python-oracledb Thick mode, the output will be:

python-oracledb thk : 1.0.0

Note that you may not see these values if you have changed the default of the :attr:`oracledb.defaults.driver_name <defaults.driver_name>` attribute or the driver_name parameter in :meth:`oracledb.init_oracle_client()`.

Database Views

This section shows some sample column values for database views. Other views also contain useful information, such as the :ref:`drcp` views discussed in :ref:`monitoringdrcp`.

V$SESSION_CONNECT_INFO

The following table lists sample values for some V$SESSION_CONNECT_INFO columns:

.. list-table-with-summary:: Sample V$SESSION_CONNECT_INFO column values
    :header-rows: 1
    :class: wy-table-responsive
    :widths: 15 10 10
    :name: V$SESSION_CONNECT_INFO
    :summary: The first column is the name of V$SESSION_CONNECT_INFO view's column. The second column lists a sample python-oracledb Thick mode value. The third column list a sample python-oracledb Thin mode value.

    * - Column
      - Thick value
      - Thin value
    * - CLIENT_OCI_LIBRARY
      - The Oracle Client or Instant Client type, such as "Full Instant Client"
      - "Unknown"
    * - CLIENT_VERSION
      - The Oracle Client library version number
      - "1.0.0.0.0" (the python-oracledb version number with an extra .0.0)
    * - CLIENT_DRIVER
      - "python-oracledb thk : 1.0.0"
      - "python-oracledb thn : 1.0.0"


V$SESSION

The following table list sample values for columns with differences in V$SESSION.

.. list-table-with-summary:: Sample V$SESSION column values
    :header-rows: 1
    :class: wy-table-responsive
    :widths: 15 10 10
    :name: V$SESSION_COLUMN_VALUES
    :summary: The first column is the name of the column. The second column lists a sample python-oracledb Thick mode value. The third column lists a sample python-oracledb Thin mode value.

    * - Column
      - Thick value
      - Thin value
    * - TERMINAL
      - similar to `ttys001`
      - the string "unknown"
    * - PROGRAM
      - similar to `python@myuser-mac2 (TNS V1-V3)`
      - the contents of Python's ``sys.executable``, such as `/Users/myuser/.pyenv/versions/3.9.6/bin/python`
    * - MODULE
      - similar to `python@myuser-mac2 (TNS V1-V3)`
      - the contents of Python's ``sys.executable``, such as `/Users/myuser/.pyenv/versions/3.9.6/bin/python`

The MODULE column value can be set as shown in :ref:`endtoendtracing`.

Low Level Python-oracledb Driver Tracing

Low level tracing is mostly useful to maintainers of python-oracledb.

  • For the python-oracledb Thin mode, packets can be traced by setting the environment variable:

    PYO_DEBUG_PACKETS=1
    

    Output goes to stdout. The logging is similar to an Oracle Net trace of level 16.

  • The python-oracledb Thick mode can be traced using: