Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions instana/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ def boot_agent():
from .instrumentation.tornado import server
from .instrumentation import logging
from .instrumentation import pymysql
from .instrumentation import psycopg2
from .instrumentation import redis
from .instrumentation import sqlalchemy
from .instrumentation import sudsjurko
Expand Down
21 changes: 13 additions & 8 deletions instana/instrumentation/pep0249.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,9 +20,14 @@ def __init__(self, cursor, module_name,
def _collect_kvs(self, span, sql):
try:
span.set_tag(ext.SPAN_KIND, 'exit')
span.set_tag(ext.DATABASE_INSTANCE, self._connect_params[1]['db'])

if 'db' in self._connect_params[1]:
span.set_tag(ext.DATABASE_INSTANCE, self._connect_params[1]['db'])
elif 'database' in self._connect_params[1]:
span.set_tag(ext.DATABASE_INSTANCE, self._connect_params[1]['database'])

span.set_tag(ext.DATABASE_STATEMENT, sql_sanitizer(sql))
span.set_tag(ext.DATABASE_TYPE, 'mysql')
# span.set_tag(ext.DATABASE_TYPE, 'mysql')
span.set_tag(ext.DATABASE_USER, self._connect_params[1]['user'])
span.set_tag('host', "%s:%s" %
(self._connect_params[1]['host'],
Expand All @@ -35,8 +40,8 @@ def _collect_kvs(self, span, sql):
def execute(self, sql, params=None):
parent_span = tracer.active_span

# If we're not tracing, just return
if parent_span is None:
# If not tracing or we're being called from sqlalchemy, just pass through
if (parent_span is None) or (parent_span.operation_name == "sqlalchemy"):
return self.__wrapped__.execute(sql, params)

with tracer.start_active_span(self._module_name, child_of=parent_span) as scope:
Expand All @@ -54,8 +59,8 @@ def execute(self, sql, params=None):
def executemany(self, sql, seq_of_parameters):
parent_span = tracer.active_span

# If we're not tracing, just return
if parent_span is None:
# If not tracing or we're being called from sqlalchemy, just pass through
if (parent_span is None) or (parent_span.operation_name == "sqlalchemy"):
return self.__wrapped__.executemany(sql, seq_of_parameters)

with tracer.start_active_span(self._module_name, child_of=parent_span) as scope:
Expand All @@ -73,8 +78,8 @@ def executemany(self, sql, seq_of_parameters):
def callproc(self, proc_name, params):
parent_span = tracer.active_span

# If we're not tracing, just return
if parent_span is None:
# If not tracing or we're being called from sqlalchemy, just pass through
if (parent_span is None) or (parent_span.operation_name == "sqlalchemy"):
return self.__wrapped__.execute(proc_name, params)

with tracer.start_active_span(self._module_name, child_of=parent_span) as scope:
Expand Down
30 changes: 30 additions & 0 deletions instana/instrumentation/psycopg2.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
from __future__ import absolute_import

import copy
import wrapt

from ..log import logger
from .pep0249 import ConnectionFactory

try:
import psycopg2
import psycopg2.extras

cf = ConnectionFactory(connect_func=psycopg2.connect, module_name='postgres')

setattr(psycopg2, 'connect', cf)
if hasattr(psycopg2, 'Connect'):
setattr(psycopg2, 'Connect', cf)

@wrapt.patch_function_wrapper('psycopg2.extensions', 'register_type')
def register_type_with_instana(wrapped, instance, args, kwargs):
args_clone = list(copy.copy(args))

if hasattr(args_clone[1], '__wrapped__'):
args_clone[1] = args_clone[1].__wrapped__

return wrapped(*args_clone, **kwargs)

logger.debug("Instrumenting psycopg2")
except ImportError:
pass
10 changes: 10 additions & 0 deletions instana/json_span.py
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ class Data(BaseSpan):
custom = None
http = None
log = None
pg = None
rabbitmq = None
redis = None
rpc = None
Expand Down Expand Up @@ -74,6 +75,15 @@ class MySQLData(BaseSpan):
error = None


class PostgresData(BaseSpan):
db = None
host = None
port = None
user = None
stmt = None
error = None


class RabbitmqData(BaseSpan):
exchange = None
queue = None
Expand Down
16 changes: 13 additions & 3 deletions instana/recorder.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@

import instana.singletons

from .json_span import (CustomData, Data, HttpData, JsonSpan, LogData, MySQLData,
from .json_span import (CustomData, Data, HttpData, JsonSpan, LogData, MySQLData, PostgresData,
RabbitmqData, RedisData, RenderData, RPCData, SDKData, SoapData,
SQLAlchemyData)
from .log import logger
Expand All @@ -24,12 +24,12 @@
class InstanaRecorder(SpanRecorder):
THREAD_NAME = "Instana Span Reporting"
registered_spans = ("aiohttp-client", "aiohttp-server", "django", "log", "memcache", "mysql",
"rabbitmq", "redis", "render", "rpc-client", "rpc-server", "sqlalchemy", "soap",
"postgres", "rabbitmq", "redis", "render", "rpc-client", "rpc-server", "sqlalchemy", "soap",
"tornado-client", "tornado-server", "urllib3", "wsgi")
http_spans = ("aiohttp-client", "aiohttp-server", "django", "http", "soap", "tornado-client",
"tornado-server", "urllib3", "wsgi")

exit_spans = ("aiohttp-client", "log", "memcache", "mysql", "rabbitmq", "redis", "rpc-client",
exit_spans = ("aiohttp-client", "log", "memcache", "mysql", "postgres", "rabbitmq", "redis", "rpc-client",
"sqlalchemy", "soap", "tornado-client", "urllib3")
entry_spans = ("aiohttp-server", "django", "wsgi", "rabbitmq", "rpc-server", "tornado-server")
local_spans = ("log", "render")
Expand Down Expand Up @@ -202,6 +202,16 @@ def build_registered_span(self, span):
tskey = list(data.custom.logs.keys())[0]
data.mysql.error = data.custom.logs[tskey]['message']

if span.operation_name == "postgres":
data.pg = PostgresData(host=span.tags.pop('host', None),
db=span.tags.pop(ext.DATABASE_INSTANCE, None),
user=span.tags.pop(ext.DATABASE_USER, None),
stmt=span.tags.pop(ext.DATABASE_STATEMENT, None),
error=span.tags.pop('pg.error', None))
if (data.custom is not None) and (data.custom.logs is not None) and len(data.custom.logs):
tskey = list(data.custom.logs.keys())[0]
data.pg.error = data.custom.logs[tskey]['message']

if span.operation_name == "log":
data.log = {}
# use last special key values
Expand Down
2 changes: 1 addition & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,10 +76,10 @@ def check_setuptools():
'mock>=2.0.0',
'mysqlclient>=1.3.14;python_version>="3.5"',
'MySQL-python>=1.2.5;python_version<="2.7"',
'psycopg2>=2.7.1',
'PyMySQL[rsa]>=0.9.1',
'pyOpenSSL>=16.1.0;python_version<="2.7"',
'pytest>=3.0.1',
'psycopg2>=2.7.1',
'redis<3.0.0',
'requests>=2.17.1',
'sqlalchemy>=1.1.15',
Expand Down
17 changes: 3 additions & 14 deletions tests/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,23 +21,12 @@
"""
PostgreSQL Environment
"""
if 'POSTGRESQL_HOST' in os.environ:
testenv['postgresql_host']= os.environ['POSTGRESQL_HOST']
elif 'TRAVIS_POSTGRESQL_HOST' in os.environ:
testenv['postgresql_host'] = os.environ['TRAVIS_POSTGRESQL_HOST']
else:
testenv['postgresql_host'] = '127.0.0.1'

testenv['postgresql_port'] = int(os.environ.get('POSTGRESQL_PORT', '3306'))
testenv['postgresql_host'] = os.environ.get('POSTGRESQL_HOST', '127.0.0.1')
testenv['postgresql_port'] = int(os.environ.get('POSTGRESQL_PORT', '5432'))
testenv['postgresql_db'] = os.environ.get('POSTGRESQL_DB', 'circle_test')
testenv['postgresql_user'] = os.environ.get('POSTGRESQL_USER', 'root')
testenv['postgresql_pw'] = os.environ.get('POSTGRESQL_PW', '')

if 'POSTGRESQL_PW' in os.environ:
testenv['postgresql_pw'] = os.environ['POSTGRESQL_PW']
elif 'TRAVIS_POSTGRESQL_PASS' in os.environ:
testenv['postgresql_pw'] = os.environ['TRAVIS_POSTGRESQL_PASS']
else:
testenv['postgresql_pw'] = ''

"""
Redis Environment
Expand Down
Loading