1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
|
"""
Private utilities for testing
"""
import _awscrt
from awscrt import NativeResource
from awscrt.common import join_all_native_threads
import gc
import inspect
import os
import sys
import time
import types
from awscrt.io import ClientBootstrap, DefaultHostResolver, EventLoopGroup
def native_memory_usage() -> int:
"""
Returns number of bytes currently allocated by awscrt's native code.
`AWS_CRT_MEMORY_TRACING `environment variable must be set before module
is loaded, or 0 will always be returned. Legal values are:
* `AWS_CRT_MEMORY_TRACING=0`: No tracing
* `AWS_CRT_MEMORY_TRACING=1`: Only track allocation sizes and total allocated
* `AWS_CRT_MEMORY_TRACING=2`: Capture callstacks for each allocation
"""
return _awscrt.native_memory_usage()
def dump_native_memory():
"""
If there are outstanding allocations from awscrt's native code, dump them
to log, along with any information gathered based on the tracing level.
In order to see the dump, logging must initialized at `LogLevel.Trace`
and the `AWS_CRT_MEMORY_TRACING` environment variable must be non-zero
when module is loaded. Legal values are:
* `AWS_CRT_MEMORY_TRACING=0`: No tracing
* `AWS_CRT_MEMORY_TRACING=1`: Only track allocation sizes and total allocated
* `AWS_CRT_MEMORY_TRACING=2`: Capture callstacks for each allocation
"""
return _awscrt.native_memory_dump()
def check_for_leaks(*, timeout_sec=10.0):
"""
Checks that all awscrt resources have been freed after a test.
If any resources still exist, debugging info is printed and an exception is raised.
Requirements:
* `awscrt.NativeResource._track_lifetime = True`: must be set before test begins
to ensure accurate tracking.
* `AWS_CRT_MEMORY_TRACING=2`: environment variable that must be set before
any awscrt modules are imported, to ensure accurate native leak checks.
* `AWS_CRT_MEMORY_PRINT_SECRETS_OK=1`: optional environment variable that
will print the full contents of leaked python objects. DO NOT SET THIS
if the test results will be made public as it may result in secrets
being leaked.
"""
ClientBootstrap.release_static_default()
EventLoopGroup.release_static_default()
DefaultHostResolver.release_static_default()
if os.getenv('AWS_CRT_MEMORY_TRACING') != '2':
raise RuntimeError("environment variable AWS_CRT_MEMORY_TRACING=2 must be set for accurate leak checks")
if not NativeResource._track_lifetime:
raise RuntimeError("awscrt.NativeResource._track_lifetime=True must be set for accurate leak checks")
# Native resources might need a few more ticks to finish cleaning themselves up.
wait_until = time.time() + timeout_sec
while time.time() < wait_until:
if not NativeResource._living and not native_memory_usage() > 0:
return
gc.collect()
# join_all_native_threads() is sometimes required to get mem usage to 0
join_all_native_threads(timeout_sec=0.1)
time.sleep(0.1)
# Print out debugging info on leaking resources
num_living_resources = len(NativeResource._living)
if num_living_resources:
leak_secrets_ok = os.getenv('AWS_CRT_MEMORY_PRINT_SECRETS_OK') == '1'
if leak_secrets_ok:
print("Leaking NativeResources:")
else:
print("Leaking NativeResources (set AWS_CRT_MEMORY_PRINT_SECRETS_OK=1 env var for more detailed report):")
def _printobj(prefix, obj):
# be sure not to accidentally print a dictionary with a password in it
if leak_secrets_ok:
s = str(obj)
if len(s) > 1000:
s = s[:1000] + '...TRUNCATED PRINT'
print(prefix, s)
else:
print(prefix, type(obj))
for i in NativeResource._living:
_printobj('-', i)
# getrefcount(i) returns 4+ here, but 2 of those are due to debugging.
# Don't show:
# - 1 for WeakSet iterator due to this for-loop.
# - 1 for getrefcount(i)'s reference.
# But do show:
# - 1 for item's self-reference.
# - the rest are what's causing this leak.
refcount = sys.getrefcount(i) - 2
# Gather list of referrers, but don't show those created by the act of iterating the WeakSet
referrers = []
for r in gc.get_referrers(i):
if isinstance(r, types.FrameType):
frameinfo = inspect.getframeinfo(r)
our_fault = (frameinfo.filename.endswith('_weakrefset.py') or
frameinfo.filename.endswith('awscrt/_test.py'))
if our_fault:
continue
referrers.append(r)
print(' sys.getrefcount():', refcount)
print(' gc.referrers():', len(referrers))
for r in referrers:
if isinstance(r, types.FrameType):
_printobj(' -', inspect.getframeinfo(r))
else:
_printobj(' -', r)
mem_bytes = native_memory_usage()
if mem_bytes > 0:
print('Leaking {} bytes native memory (enable Trace logging to see more)'.format(mem_bytes))
dump_native_memory()
raise RuntimeError("awscrt leak check failed. {} NativeResource objects. {} bytes native memory".format(
num_living_resources, mem_bytes))
|