Skip to content

Commit e891528

Browse files
authored
Capture module versions (#588)
* Change n is None to not n None types are falsey so we can shorten this expression to `if not module`. * Use in instead of .find `in` is more performant than find for search a string so use this instead. * Simplify and combine sub path module logic Do not include module.sub_paths as separate modules. Skip these except for `newrelic.hooks`. * Exclude standard lib/built-in modules Previously, we were capturing standard library and built-in Python modules as plugins. These are included with the version of Python the user installed and are not packages that need to be captured so exclude these from the list. * Capture module versions * Fixup: remove pkg_resources check * Ignore pylint function-redefined * Check plugin version info in tests
1 parent a81513d commit e891528

File tree

2 files changed

+39
-27
lines changed

2 files changed

+39
-27
lines changed

newrelic/core/environment.py

+31-21
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
import os
2121
import platform
2222
import sys
23+
import sysconfig
2324

2425
import newrelic
2526
from newrelic.common.system_info import (
@@ -178,41 +179,50 @@ def environment_settings():
178179
env.extend(dispatcher)
179180

180181
# Module information.
182+
purelib = sysconfig.get_path("purelib")
183+
platlib = sysconfig.get_path("platlib")
181184

182185
plugins = []
183186

187+
get_version = None
188+
# importlib was introduced into the standard library starting in Python3.8.
189+
if "importlib" in sys.modules and hasattr(sys.modules["importlib"], "metadata"):
190+
get_version = sys.modules["importlib"].metadata.version
191+
elif "pkg_resources" in sys.modules:
192+
193+
def get_version(name): # pylint: disable=function-redefined
194+
return sys.modules["pkg_resources"].get_distribution(name).version
195+
184196
# Using any iterable to create a snapshot of sys.modules can occassionally
185197
# fail in a rare case when modules are imported in parallel by different
186198
# threads.
187199
#
188200
# TL;DR: Do NOT use an iterable on the original sys.modules to generate the
189201
# list
190-
191202
for name, module in sys.modules.copy().items():
203+
# Exclude lib.sub_paths as independent modules except for newrelic.hooks.
204+
if "." in name and not name.startswith("newrelic.hooks."):
205+
continue
192206
# If the module isn't actually loaded (such as failed relative imports
193207
# in Python 2.7), the module will be None and should not be reported.
194-
if module is None:
208+
if not module:
209+
continue
210+
# Exclude standard library/built-in modules.
211+
# Third-party modules can be installed in either purelib or platlib directories.
212+
# See https://docs.python.org/3/library/sysconfig.html#installation-paths.
213+
if (
214+
not hasattr(module, "__file__")
215+
or not module.__file__
216+
or not module.__file__.startswith(purelib)
217+
or not module.__file__.startswith(platlib)
218+
):
195219
continue
196220

197-
if name.startswith("newrelic.hooks."):
198-
plugins.append(name)
199-
200-
elif name.find(".") == -1 and hasattr(module, "__file__"):
201-
# XXX This is disabled as it can cause notable overhead in
202-
# pathalogical cases. Will be replaced with a new system
203-
# where have a allowlist of packages we really want version
204-
# information for and will work out on case by case basis
205-
# how to extract that from the modules themselves.
206-
207-
# try:
208-
# if 'pkg_resources' in sys.modules:
209-
# version = pkg_resources.get_distribution(name).version
210-
# if version:
211-
# name = '%s (%s)' % (name, version)
212-
# except Exception:
213-
# pass
214-
215-
plugins.append(name)
221+
try:
222+
version = get_version(name)
223+
plugins.append("%s (%s)" % (name, version))
224+
except Exception:
225+
pass
216226

217227
env.append(("Plugin List", plugins))
218228

tests/agent_unittests/test_environment.py

+8-6
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,10 @@
1212
# See the License for the specific language governing permissions and
1313
# limitations under the License.
1414

15-
import pytest
1615
import sys
16+
17+
import pytest
18+
1719
from newrelic.core.environment import environment_settings
1820

1921

@@ -29,7 +31,7 @@ class Module(object):
2931

3032
def test_plugin_list():
3133
# Let's pretend we fired an import hook
32-
import newrelic.hooks.adapter_gunicorn
34+
import newrelic.hooks.adapter_gunicorn # noqa: F401
3335

3436
environment_info = environment_settings()
3537

@@ -41,6 +43,8 @@ def test_plugin_list():
4143

4244
# Check that bogus plugins don't get reported
4345
assert "newrelic.hooks.newrelic" not in plugin_list
46+
# Check that plugin that should get reported has version info.
47+
assert "pytest (%s)" % (pytest.__version__) in plugin_list
4448

4549

4650
class NoIteratorDict(object):
@@ -62,7 +66,7 @@ def __contains__(self, *args, **kwargs):
6266

6367
def test_plugin_list_uses_no_sys_modules_iterator(monkeypatch):
6468
modules = NoIteratorDict(sys.modules)
65-
monkeypatch.setattr(sys, 'modules', modules)
69+
monkeypatch.setattr(sys, "modules", modules)
6670

6771
# If environment_settings iterates over sys.modules, an attribute error will be generated
6872
environment_info = environment_settings()
@@ -113,9 +117,7 @@ def test_plugin_list_uses_no_sys_modules_iterator(monkeypatch):
113117
),
114118
),
115119
)
116-
def test_uvicorn_dispatcher(
117-
monkeypatch, loaded_modules, dispatcher, dispatcher_version, worker_version
118-
):
120+
def test_uvicorn_dispatcher(monkeypatch, loaded_modules, dispatcher, dispatcher_version, worker_version):
119121
# Let's pretend we load some modules
120122
for name, module in loaded_modules.items():
121123
monkeypatch.setitem(sys.modules, name, module)

0 commit comments

Comments
 (0)