mirror of
https://github.com/open-telemetry/opentelemetry-python-contrib.git
synced 2025-07-30 05:32:30 +08:00
Merge branch 'master' into req
This commit is contained in:
@ -2,6 +2,9 @@
|
|||||||
|
|
||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
|
- Update sqlalchemy instrumentation to follow semantic conventions
|
||||||
|
([#202](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/202))
|
||||||
|
|
||||||
## Version 0.13b0
|
## Version 0.13b0
|
||||||
|
|
||||||
Released 2020-09-17
|
Released 2020-09-17
|
||||||
|
@ -34,7 +34,6 @@ Usage
|
|||||||
engine = create_engine("sqlite:///:memory:")
|
engine = create_engine("sqlite:///:memory:")
|
||||||
SQLAlchemyInstrumentor().instrument(
|
SQLAlchemyInstrumentor().instrument(
|
||||||
engine=engine,
|
engine=engine,
|
||||||
service="service-A",
|
|
||||||
)
|
)
|
||||||
|
|
||||||
API
|
API
|
||||||
@ -66,7 +65,6 @@ class SQLAlchemyInstrumentor(BaseInstrumentor):
|
|||||||
**kwargs: Optional arguments
|
**kwargs: Optional arguments
|
||||||
``engine``: a SQLAlchemy engine instance
|
``engine``: a SQLAlchemy engine instance
|
||||||
``tracer_provider``: a TracerProvider, defaults to global
|
``tracer_provider``: a TracerProvider, defaults to global
|
||||||
``service``: the name of the service to trace.
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
An instrumented engine if passed in as an argument, None otherwise.
|
An instrumented engine if passed in as an argument, None otherwise.
|
||||||
@ -78,7 +76,6 @@ class SQLAlchemyInstrumentor(BaseInstrumentor):
|
|||||||
_get_tracer(
|
_get_tracer(
|
||||||
kwargs.get("engine"), kwargs.get("tracer_provider")
|
kwargs.get("engine"), kwargs.get("tracer_provider")
|
||||||
),
|
),
|
||||||
kwargs.get("service"),
|
|
||||||
kwargs.get("engine"),
|
kwargs.get("engine"),
|
||||||
)
|
)
|
||||||
return None
|
return None
|
||||||
|
@ -24,10 +24,9 @@ _HOST = "net.peer.name"
|
|||||||
_PORT = "net.peer.port"
|
_PORT = "net.peer.port"
|
||||||
# Database semantic conventions here:
|
# Database semantic conventions here:
|
||||||
# https://github.com/open-telemetry/opentelemetry-specification/blob/master/specification/trace/semantic_conventions/database.md
|
# https://github.com/open-telemetry/opentelemetry-specification/blob/master/specification/trace/semantic_conventions/database.md
|
||||||
_ROWS = "sql.rows" # number of rows returned by a query
|
|
||||||
_STMT = "db.statement"
|
_STMT = "db.statement"
|
||||||
_DB = "db.type"
|
_DB = "db.name"
|
||||||
_URL = "db.url"
|
_USER = "db.user"
|
||||||
|
|
||||||
|
|
||||||
def _normalize_vendor(vendor):
|
def _normalize_vendor(vendor):
|
||||||
@ -39,7 +38,7 @@ def _normalize_vendor(vendor):
|
|||||||
return "sqlite"
|
return "sqlite"
|
||||||
|
|
||||||
if "postgres" in vendor or vendor == "psycopg2":
|
if "postgres" in vendor or vendor == "psycopg2":
|
||||||
return "postgres"
|
return "postgresql"
|
||||||
|
|
||||||
return vendor
|
return vendor
|
||||||
|
|
||||||
@ -58,17 +57,15 @@ def _wrap_create_engine(func, module, args, kwargs):
|
|||||||
object that will listen to SQLAlchemy events.
|
object that will listen to SQLAlchemy events.
|
||||||
"""
|
"""
|
||||||
engine = func(*args, **kwargs)
|
engine = func(*args, **kwargs)
|
||||||
EngineTracer(_get_tracer(engine), None, engine)
|
EngineTracer(_get_tracer(engine), engine)
|
||||||
return engine
|
return engine
|
||||||
|
|
||||||
|
|
||||||
class EngineTracer:
|
class EngineTracer:
|
||||||
def __init__(self, tracer, service, engine):
|
def __init__(self, tracer, engine):
|
||||||
self.tracer = tracer
|
self.tracer = tracer
|
||||||
self.engine = engine
|
self.engine = engine
|
||||||
self.vendor = _normalize_vendor(engine.name)
|
self.vendor = _normalize_vendor(engine.name)
|
||||||
self.service = service or self.vendor
|
|
||||||
self.name = "%s.query" % self.vendor
|
|
||||||
self.current_span = None
|
self.current_span = None
|
||||||
|
|
||||||
listen(engine, "before_cursor_execute", self._before_cur_exec)
|
listen(engine, "before_cursor_execute", self._before_cur_exec)
|
||||||
@ -77,11 +74,11 @@ class EngineTracer:
|
|||||||
|
|
||||||
# pylint: disable=unused-argument
|
# pylint: disable=unused-argument
|
||||||
def _before_cur_exec(self, conn, cursor, statement, *args):
|
def _before_cur_exec(self, conn, cursor, statement, *args):
|
||||||
self.current_span = self.tracer.start_span(self.name)
|
self.current_span = self.tracer.start_span(statement)
|
||||||
with self.tracer.use_span(self.current_span, end_on_exit=False):
|
with self.tracer.use_span(self.current_span, end_on_exit=False):
|
||||||
if self.current_span.is_recording():
|
if self.current_span.is_recording():
|
||||||
self.current_span.set_attribute("service", self.vendor)
|
|
||||||
self.current_span.set_attribute(_STMT, statement)
|
self.current_span.set_attribute(_STMT, statement)
|
||||||
|
self.current_span.set_attribute("db.system", self.vendor)
|
||||||
|
|
||||||
if not _set_attributes_from_url(
|
if not _set_attributes_from_url(
|
||||||
self.current_span, conn.engine.url
|
self.current_span, conn.engine.url
|
||||||
@ -94,16 +91,7 @@ class EngineTracer:
|
|||||||
def _after_cur_exec(self, conn, cursor, statement, *args):
|
def _after_cur_exec(self, conn, cursor, statement, *args):
|
||||||
if self.current_span is None:
|
if self.current_span is None:
|
||||||
return
|
return
|
||||||
|
self.current_span.end()
|
||||||
try:
|
|
||||||
if (
|
|
||||||
cursor
|
|
||||||
and cursor.rowcount >= 0
|
|
||||||
and self.current_span.is_recording()
|
|
||||||
):
|
|
||||||
self.current_span.set_attribute(_ROWS, cursor.rowcount)
|
|
||||||
finally:
|
|
||||||
self.current_span.end()
|
|
||||||
|
|
||||||
def _handle_error(self, context):
|
def _handle_error(self, context):
|
||||||
if self.current_span is None:
|
if self.current_span is None:
|
||||||
@ -127,6 +115,8 @@ def _set_attributes_from_url(span: trace.Span, url):
|
|||||||
span.set_attribute(_PORT, url.port)
|
span.set_attribute(_PORT, url.port)
|
||||||
if url.database:
|
if url.database:
|
||||||
span.set_attribute(_DB, url.database)
|
span.set_attribute(_DB, url.database)
|
||||||
|
if url.username:
|
||||||
|
span.set_attribute(_USER, url.username)
|
||||||
|
|
||||||
return bool(url.host)
|
return bool(url.host)
|
||||||
|
|
||||||
@ -135,7 +125,7 @@ def _set_attributes_from_cursor(span: trace.Span, vendor, cursor):
|
|||||||
"""Attempt to set db connection attributes by introspecting the cursor."""
|
"""Attempt to set db connection attributes by introspecting the cursor."""
|
||||||
if not span.is_recording():
|
if not span.is_recording():
|
||||||
return
|
return
|
||||||
if vendor == "postgres":
|
if vendor == "postgresql":
|
||||||
# pylint: disable=import-outside-toplevel
|
# pylint: disable=import-outside-toplevel
|
||||||
from psycopg2.extensions import parse_dsn
|
from psycopg2.extensions import parse_dsn
|
||||||
|
|
||||||
|
@ -27,16 +27,14 @@ class TestSqlalchemyInstrumentation(TestBase):
|
|||||||
def test_trace_integration(self):
|
def test_trace_integration(self):
|
||||||
engine = create_engine("sqlite:///:memory:")
|
engine = create_engine("sqlite:///:memory:")
|
||||||
SQLAlchemyInstrumentor().instrument(
|
SQLAlchemyInstrumentor().instrument(
|
||||||
engine=engine,
|
engine=engine, tracer_provider=self.tracer_provider,
|
||||||
tracer_provider=self.tracer_provider,
|
|
||||||
service="my-database",
|
|
||||||
)
|
)
|
||||||
cnx = engine.connect()
|
cnx = engine.connect()
|
||||||
cnx.execute("SELECT 1 + 1;").fetchall()
|
cnx.execute("SELECT 1 + 1;").fetchall()
|
||||||
spans = self.memory_exporter.get_finished_spans()
|
spans = self.memory_exporter.get_finished_spans()
|
||||||
|
|
||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
self.assertEqual(spans[0].name, "sqlite.query")
|
self.assertEqual(spans[0].name, "SELECT 1 + 1;")
|
||||||
|
|
||||||
def test_not_recording(self):
|
def test_not_recording(self):
|
||||||
mock_tracer = mock.Mock()
|
mock_tracer = mock.Mock()
|
||||||
@ -49,9 +47,7 @@ class TestSqlalchemyInstrumentation(TestBase):
|
|||||||
tracer.return_value = mock_tracer
|
tracer.return_value = mock_tracer
|
||||||
engine = create_engine("sqlite:///:memory:")
|
engine = create_engine("sqlite:///:memory:")
|
||||||
SQLAlchemyInstrumentor().instrument(
|
SQLAlchemyInstrumentor().instrument(
|
||||||
engine=engine,
|
engine=engine, tracer_provider=self.tracer_provider,
|
||||||
tracer_provider=self.tracer_provider,
|
|
||||||
service="my-database",
|
|
||||||
)
|
)
|
||||||
cnx = engine.connect()
|
cnx = engine.connect()
|
||||||
cnx.execute("SELECT 1 + 1;").fetchall()
|
cnx.execute("SELECT 1 + 1;").fetchall()
|
||||||
@ -70,4 +66,4 @@ class TestSqlalchemyInstrumentation(TestBase):
|
|||||||
spans = self.memory_exporter.get_finished_spans()
|
spans = self.memory_exporter.get_finished_spans()
|
||||||
|
|
||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
self.assertEqual(spans[0].name, "sqlite.query")
|
self.assertEqual(spans[0].name, "SELECT 1 + 1;")
|
||||||
|
@ -20,7 +20,7 @@ from sqlalchemy.orm import sessionmaker
|
|||||||
|
|
||||||
from opentelemetry import trace
|
from opentelemetry import trace
|
||||||
from opentelemetry.instrumentation.sqlalchemy import SQLAlchemyInstrumentor
|
from opentelemetry.instrumentation.sqlalchemy import SQLAlchemyInstrumentor
|
||||||
from opentelemetry.instrumentation.sqlalchemy.engine import _DB, _ROWS, _STMT
|
from opentelemetry.instrumentation.sqlalchemy.engine import _DB, _STMT
|
||||||
from opentelemetry.test.test_base import TestBase
|
from opentelemetry.test.test_base import TestBase
|
||||||
|
|
||||||
Base = declarative_base()
|
Base = declarative_base()
|
||||||
@ -109,9 +109,8 @@ class SQLAlchemyTestMixin(TestBase):
|
|||||||
SQLAlchemyInstrumentor().uninstrument()
|
SQLAlchemyInstrumentor().uninstrument()
|
||||||
super().tearDown()
|
super().tearDown()
|
||||||
|
|
||||||
def _check_span(self, span):
|
def _check_span(self, span, name):
|
||||||
self.assertEqual(span.name, "{}.query".format(self.VENDOR))
|
self.assertEqual(span.name, name)
|
||||||
self.assertEqual(span.attributes.get("service"), self.SERVICE)
|
|
||||||
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
||||||
self.assertIs(span.status.status_code, trace.status.StatusCode.UNSET)
|
self.assertIs(span.status.status_code, trace.status.StatusCode.UNSET)
|
||||||
self.assertGreater((span.end_time - span.start_time), 0)
|
self.assertGreater((span.end_time - span.start_time), 0)
|
||||||
@ -125,9 +124,13 @@ class SQLAlchemyTestMixin(TestBase):
|
|||||||
spans = self.memory_exporter.get_finished_spans()
|
spans = self.memory_exporter.get_finished_spans()
|
||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
span = spans[0]
|
span = spans[0]
|
||||||
self._check_span(span)
|
stmt = "INSERT INTO players (id, name) VALUES "
|
||||||
|
if span.attributes.get("db.system") == "sqlite":
|
||||||
|
stmt += "(?, ?)"
|
||||||
|
else:
|
||||||
|
stmt += "(%(id)s, %(name)s)"
|
||||||
|
self._check_span(span, stmt)
|
||||||
self.assertIn("INSERT INTO players", span.attributes.get(_STMT))
|
self.assertIn("INSERT INTO players", span.attributes.get(_STMT))
|
||||||
self.assertEqual(span.attributes.get(_ROWS), 1)
|
|
||||||
self.check_meta(span)
|
self.check_meta(span)
|
||||||
|
|
||||||
def test_session_query(self):
|
def test_session_query(self):
|
||||||
@ -138,7 +141,12 @@ class SQLAlchemyTestMixin(TestBase):
|
|||||||
spans = self.memory_exporter.get_finished_spans()
|
spans = self.memory_exporter.get_finished_spans()
|
||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
span = spans[0]
|
span = spans[0]
|
||||||
self._check_span(span)
|
stmt = "SELECT players.id AS players_id, players.name AS players_name \nFROM players \nWHERE players.name = "
|
||||||
|
if span.attributes.get("db.system") == "sqlite":
|
||||||
|
stmt += "?"
|
||||||
|
else:
|
||||||
|
stmt += "%(name_1)s"
|
||||||
|
self._check_span(span, stmt)
|
||||||
self.assertIn(
|
self.assertIn(
|
||||||
"SELECT players.id AS players_id, players.name AS players_name \nFROM players \nWHERE players.name",
|
"SELECT players.id AS players_id, players.name AS players_name \nFROM players \nWHERE players.name",
|
||||||
span.attributes.get(_STMT),
|
span.attributes.get(_STMT),
|
||||||
@ -147,24 +155,26 @@ class SQLAlchemyTestMixin(TestBase):
|
|||||||
|
|
||||||
def test_engine_connect_execute(self):
|
def test_engine_connect_execute(self):
|
||||||
# ensures that engine.connect() is properly traced
|
# ensures that engine.connect() is properly traced
|
||||||
|
stmt = "SELECT * FROM players"
|
||||||
with self.connection() as conn:
|
with self.connection() as conn:
|
||||||
rows = conn.execute("SELECT * FROM players").fetchall()
|
rows = conn.execute(stmt).fetchall()
|
||||||
self.assertEqual(len(rows), 0)
|
self.assertEqual(len(rows), 0)
|
||||||
|
|
||||||
spans = self.memory_exporter.get_finished_spans()
|
spans = self.memory_exporter.get_finished_spans()
|
||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
span = spans[0]
|
span = spans[0]
|
||||||
self._check_span(span)
|
self._check_span(span, stmt)
|
||||||
self.assertEqual(span.attributes.get(_STMT), "SELECT * FROM players")
|
self.assertEqual(span.attributes.get(_STMT), "SELECT * FROM players")
|
||||||
self.check_meta(span)
|
self.check_meta(span)
|
||||||
|
|
||||||
def test_parent(self):
|
def test_parent(self):
|
||||||
"""Ensure that sqlalchemy works with opentelemetry."""
|
"""Ensure that sqlalchemy works with opentelemetry."""
|
||||||
|
stmt = "SELECT * FROM players"
|
||||||
tracer = self.tracer_provider.get_tracer("sqlalch_svc")
|
tracer = self.tracer_provider.get_tracer("sqlalch_svc")
|
||||||
|
|
||||||
with tracer.start_as_current_span("sqlalch_op"):
|
with tracer.start_as_current_span("sqlalch_op"):
|
||||||
with self.connection() as conn:
|
with self.connection() as conn:
|
||||||
rows = conn.execute("SELECT * FROM players").fetchall()
|
rows = conn.execute(stmt).fetchall()
|
||||||
self.assertEqual(len(rows), 0)
|
self.assertEqual(len(rows), 0)
|
||||||
|
|
||||||
spans = self.memory_exporter.get_finished_spans()
|
spans = self.memory_exporter.get_finished_spans()
|
||||||
@ -178,5 +188,4 @@ class SQLAlchemyTestMixin(TestBase):
|
|||||||
self.assertEqual(parent_span.name, "sqlalch_op")
|
self.assertEqual(parent_span.name, "sqlalch_op")
|
||||||
self.assertEqual(parent_span.instrumentation_info.name, "sqlalch_svc")
|
self.assertEqual(parent_span.instrumentation_info.name, "sqlalch_svc")
|
||||||
|
|
||||||
self.assertEqual(child_span.name, "{}.query".format(self.VENDOR))
|
self.assertEqual(child_span.name, stmt)
|
||||||
self.assertEqual(child_span.attributes.get("service"), self.SERVICE)
|
|
||||||
|
@ -64,7 +64,6 @@ class SQLAlchemyInstrumentTestCase(TestBase):
|
|||||||
self.assertEqual(len(traces), 1)
|
self.assertEqual(len(traces), 1)
|
||||||
span = traces[0]
|
span = traces[0]
|
||||||
# check subset of span fields
|
# check subset of span fields
|
||||||
self.assertEqual(span.name, "postgres.query")
|
self.assertEqual(span.name, "SELECT 1")
|
||||||
self.assertEqual(span.attributes.get("service"), "postgres")
|
|
||||||
self.assertIs(span.status.status_code, trace.status.StatusCode.UNSET)
|
self.assertIs(span.status.status_code, trace.status.StatusCode.UNSET)
|
||||||
self.assertGreater((span.end_time - span.start_time), 0)
|
self.assertGreater((span.end_time - span.start_time), 0)
|
||||||
|
@ -23,8 +23,8 @@ from opentelemetry.instrumentation.sqlalchemy.engine import (
|
|||||||
_DB,
|
_DB,
|
||||||
_HOST,
|
_HOST,
|
||||||
_PORT,
|
_PORT,
|
||||||
_ROWS,
|
|
||||||
_STMT,
|
_STMT,
|
||||||
|
_USER,
|
||||||
)
|
)
|
||||||
|
|
||||||
from .mixins import SQLAlchemyTestMixin
|
from .mixins import SQLAlchemyTestMixin
|
||||||
@ -45,7 +45,6 @@ class MysqlConnectorTestCase(SQLAlchemyTestMixin):
|
|||||||
|
|
||||||
VENDOR = "mysql"
|
VENDOR = "mysql"
|
||||||
SQL_DB = "opentelemetry-tests"
|
SQL_DB = "opentelemetry-tests"
|
||||||
SERVICE = "mysql"
|
|
||||||
ENGINE_ARGS = {
|
ENGINE_ARGS = {
|
||||||
"url": "mysql+mysqlconnector://%(user)s:%(password)s@%(host)s:%(port)s/%(database)s"
|
"url": "mysql+mysqlconnector://%(user)s:%(password)s@%(host)s:%(port)s/%(database)s"
|
||||||
% MYSQL_CONFIG
|
% MYSQL_CONFIG
|
||||||
@ -55,6 +54,8 @@ class MysqlConnectorTestCase(SQLAlchemyTestMixin):
|
|||||||
# check database connection tags
|
# check database connection tags
|
||||||
self.assertEqual(span.attributes.get(_HOST), MYSQL_CONFIG["host"])
|
self.assertEqual(span.attributes.get(_HOST), MYSQL_CONFIG["host"])
|
||||||
self.assertEqual(span.attributes.get(_PORT), MYSQL_CONFIG["port"])
|
self.assertEqual(span.attributes.get(_PORT), MYSQL_CONFIG["port"])
|
||||||
|
self.assertEqual(span.attributes.get(_DB), MYSQL_CONFIG["database"])
|
||||||
|
self.assertEqual(span.attributes.get(_USER), MYSQL_CONFIG["user"])
|
||||||
|
|
||||||
def test_engine_execute_errors(self):
|
def test_engine_execute_errors(self):
|
||||||
# ensures that SQL errors are reported
|
# ensures that SQL errors are reported
|
||||||
@ -66,13 +67,11 @@ class MysqlConnectorTestCase(SQLAlchemyTestMixin):
|
|||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
span = spans[0]
|
span = spans[0]
|
||||||
# span fields
|
# span fields
|
||||||
self.assertEqual(span.name, "{}.query".format(self.VENDOR))
|
self.assertEqual(span.name, "SELECT * FROM a_wrong_table")
|
||||||
self.assertEqual(span.attributes.get("service"), self.SERVICE)
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
span.attributes.get(_STMT), "SELECT * FROM a_wrong_table"
|
span.attributes.get(_STMT), "SELECT * FROM a_wrong_table"
|
||||||
)
|
)
|
||||||
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
||||||
self.assertIsNone(span.attributes.get(_ROWS))
|
|
||||||
self.check_meta(span)
|
self.check_meta(span)
|
||||||
self.assertTrue(span.end_time - span.start_time > 0)
|
self.assertTrue(span.end_time - span.start_time > 0)
|
||||||
# check the error
|
# check the error
|
||||||
|
@ -24,7 +24,6 @@ from opentelemetry.instrumentation.sqlalchemy.engine import (
|
|||||||
_DB,
|
_DB,
|
||||||
_HOST,
|
_HOST,
|
||||||
_PORT,
|
_PORT,
|
||||||
_ROWS,
|
|
||||||
_STMT,
|
_STMT,
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -44,9 +43,8 @@ class PostgresTestCase(SQLAlchemyTestMixin):
|
|||||||
|
|
||||||
__test__ = True
|
__test__ = True
|
||||||
|
|
||||||
VENDOR = "postgres"
|
VENDOR = "postgresql"
|
||||||
SQL_DB = "opentelemetry-tests"
|
SQL_DB = "opentelemetry-tests"
|
||||||
SERVICE = "postgres"
|
|
||||||
ENGINE_ARGS = {
|
ENGINE_ARGS = {
|
||||||
"url": "postgresql://%(user)s:%(password)s@%(host)s:%(port)s/%(dbname)s"
|
"url": "postgresql://%(user)s:%(password)s@%(host)s:%(port)s/%(dbname)s"
|
||||||
% POSTGRES_CONFIG
|
% POSTGRES_CONFIG
|
||||||
@ -67,13 +65,11 @@ class PostgresTestCase(SQLAlchemyTestMixin):
|
|||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
span = spans[0]
|
span = spans[0]
|
||||||
# span fields
|
# span fields
|
||||||
self.assertEqual(span.name, "{}.query".format(self.VENDOR))
|
self.assertEqual(span.name, "SELECT * FROM a_wrong_table")
|
||||||
self.assertEqual(span.attributes.get("service"), self.SERVICE)
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
span.attributes.get(_STMT), "SELECT * FROM a_wrong_table"
|
span.attributes.get(_STMT), "SELECT * FROM a_wrong_table"
|
||||||
)
|
)
|
||||||
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
||||||
self.assertIsNone(span.attributes.get(_ROWS))
|
|
||||||
self.check_meta(span)
|
self.check_meta(span)
|
||||||
self.assertTrue(span.end_time - span.start_time > 0)
|
self.assertTrue(span.end_time - span.start_time > 0)
|
||||||
# check the error
|
# check the error
|
||||||
@ -88,9 +84,8 @@ class PostgresCreatorTestCase(PostgresTestCase):
|
|||||||
of `PostgresTestCase`, but it uses a specific `creator` function.
|
of `PostgresTestCase`, but it uses a specific `creator` function.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
VENDOR = "postgres"
|
VENDOR = "postgresql"
|
||||||
SQL_DB = "opentelemetry-tests"
|
SQL_DB = "opentelemetry-tests"
|
||||||
SERVICE = "postgres"
|
|
||||||
ENGINE_ARGS = {
|
ENGINE_ARGS = {
|
||||||
"url": "postgresql://",
|
"url": "postgresql://",
|
||||||
"creator": lambda: psycopg2.connect(**POSTGRES_CONFIG),
|
"creator": lambda: psycopg2.connect(**POSTGRES_CONFIG),
|
||||||
|
@ -18,7 +18,7 @@ import pytest
|
|||||||
from sqlalchemy.exc import OperationalError
|
from sqlalchemy.exc import OperationalError
|
||||||
|
|
||||||
from opentelemetry import trace
|
from opentelemetry import trace
|
||||||
from opentelemetry.instrumentation.sqlalchemy.engine import _DB, _ROWS, _STMT
|
from opentelemetry.instrumentation.sqlalchemy.engine import _DB, _STMT
|
||||||
|
|
||||||
from .mixins import SQLAlchemyTestMixin
|
from .mixins import SQLAlchemyTestMixin
|
||||||
|
|
||||||
@ -30,26 +30,24 @@ class SQLiteTestCase(SQLAlchemyTestMixin):
|
|||||||
|
|
||||||
VENDOR = "sqlite"
|
VENDOR = "sqlite"
|
||||||
SQL_DB = ":memory:"
|
SQL_DB = ":memory:"
|
||||||
SERVICE = "sqlite"
|
|
||||||
ENGINE_ARGS = {"url": "sqlite:///:memory:"}
|
ENGINE_ARGS = {"url": "sqlite:///:memory:"}
|
||||||
|
|
||||||
def test_engine_execute_errors(self):
|
def test_engine_execute_errors(self):
|
||||||
# ensures that SQL errors are reported
|
# ensures that SQL errors are reported
|
||||||
|
stmt = "SELECT * FROM a_wrong_table"
|
||||||
with pytest.raises(OperationalError):
|
with pytest.raises(OperationalError):
|
||||||
with self.connection() as conn:
|
with self.connection() as conn:
|
||||||
conn.execute("SELECT * FROM a_wrong_table").fetchall()
|
conn.execute(stmt).fetchall()
|
||||||
|
|
||||||
spans = self.memory_exporter.get_finished_spans()
|
spans = self.memory_exporter.get_finished_spans()
|
||||||
self.assertEqual(len(spans), 1)
|
self.assertEqual(len(spans), 1)
|
||||||
span = spans[0]
|
span = spans[0]
|
||||||
# span fields
|
# span fields
|
||||||
self.assertEqual(span.name, "{}.query".format(self.VENDOR))
|
self.assertEqual(span.name, stmt)
|
||||||
self.assertEqual(span.attributes.get("service"), self.SERVICE)
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
span.attributes.get(_STMT), "SELECT * FROM a_wrong_table"
|
span.attributes.get(_STMT), "SELECT * FROM a_wrong_table"
|
||||||
)
|
)
|
||||||
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
self.assertEqual(span.attributes.get(_DB), self.SQL_DB)
|
||||||
self.assertIsNone(span.attributes.get(_ROWS))
|
|
||||||
self.assertTrue((span.end_time - span.start_time) > 0)
|
self.assertTrue((span.end_time - span.start_time) > 0)
|
||||||
# check the error
|
# check the error
|
||||||
self.assertIs(
|
self.assertIs(
|
||||||
|
Reference in New Issue
Block a user