From f1e3865d1e0ea2213e57e717aa770756b19e7e36 Mon Sep 17 00:00:00 2001 From: mousberg Date: Thu, 27 Feb 2025 17:17:31 +0000 Subject: [PATCH 1/3] Fix memory leak in AsyncCompletions.parse() with dynamically created models This commit fixes a memory leak issue in AsyncCompletions.parse() when repeatedly called with Pydantic models created via create_model(). The issue was occurring because schema representations of models were being retained indefinitely. The fix implements a WeakKeyDictionary cache that allows the schema objects to be garbage-collected when the model types are no longer referenced elsewhere in code. Added test cases to verify the fix prevents memory leaks with dynamically created models. --- src/openai/lib/_parsing/_completions.py | 15 +++- tests/lib/_parsing/test_memory_leak.py | 101 ++++++++++++++++++++++++ 2 files changed, 115 insertions(+), 1 deletion(-) create mode 100644 tests/lib/_parsing/test_memory_leak.py diff --git a/src/openai/lib/_parsing/_completions.py b/src/openai/lib/_parsing/_completions.py index c160070b66..fc7e6e5ab8 100644 --- a/src/openai/lib/_parsing/_completions.py +++ b/src/openai/lib/_parsing/_completions.py @@ -1,6 +1,7 @@ from __future__ import annotations import json +import weakref from typing import TYPE_CHECKING, Any, Iterable, cast from typing_extensions import TypeVar, TypeGuard, assert_never @@ -28,6 +29,9 @@ from ...types.chat.completion_create_params import ResponseFormat as ResponseFormatParam from ...types.chat.chat_completion_message_tool_call import Function +# Cache to store weak references to schema objects +_schema_cache = weakref.WeakKeyDictionary() + ResponseFormatT = TypeVar( "ResponseFormatT", # if it isn't given then we don't do any parsing @@ -243,6 +247,10 @@ def type_to_response_format_param( # can only be a `type` response_format = cast(type, response_format) + # Check if we already have a schema for this type in the cache + if response_format in _schema_cache: + return _schema_cache[response_format] + json_schema_type: type[pydantic.BaseModel] | pydantic.TypeAdapter[Any] | None = None if is_basemodel_type(response_format): @@ -254,7 +262,7 @@ def type_to_response_format_param( else: raise TypeError(f"Unsupported response_format type - {response_format}") - return { + schema_param = { "type": "json_schema", "json_schema": { "schema": to_strict_json_schema(json_schema_type), @@ -262,3 +270,8 @@ def type_to_response_format_param( "strict": True, }, } + + # Store a weak reference to the schema parameter + _schema_cache[response_format] = schema_param + + return schema_param diff --git a/tests/lib/_parsing/test_memory_leak.py b/tests/lib/_parsing/test_memory_leak.py new file mode 100644 index 0000000000..3e7dd46a22 --- /dev/null +++ b/tests/lib/_parsing/test_memory_leak.py @@ -0,0 +1,101 @@ +import unittest +import gc +import sys +from unittest.mock import AsyncMock, patch, MagicMock +from typing import List + +import pytest +from pydantic import Field, create_model + +from openai.resources.beta.chat.completions import AsyncCompletions +from openai.lib._parsing import type_to_response_format_param +from openai.lib._parsing._completions import _schema_cache + +class TestMemoryLeak(unittest.TestCase): + def setUp(self): + # Clear the schema cache before each test + _schema_cache.clear() + + def test_schema_cache_with_models(self): + """Test if schema cache properly handles dynamic models and prevents memory leak""" + + StepModel = create_model( + "Step", + explanation=(str, Field()), + output=(str, Field()), + ) + + # Create several models and ensure they're cached properly + models = [] + for i in range(5): + model = create_model( + f"MathResponse{i}", + steps=(List[StepModel], Field()), + final_answer=(str, Field()), + ) + models.append(model) + + # Convert model to response format param + param = type_to_response_format_param(model) + + # Check if the model is in the cache + self.assertIn(model, _schema_cache) + + # Test that all models are in the cache + self.assertEqual(len(_schema_cache), 5) + + # Let the models go out of scope and trigger garbage collection + models = None + gc.collect() + + # After garbage collection, the cache should be empty or reduced + # since we're using weakref.WeakKeyDictionary + self.assertLess(len(_schema_cache), 5) + +@pytest.mark.asyncio +async def test_async_completions_parse_memory(): + """Test if AsyncCompletions.parse() doesn't leak memory with dynamic models""" + StepModel = create_model( + "Step", + explanation=(str, Field()), + output=(str, Field()), + ) + + # Clear the cache and record initial state + _schema_cache.clear() + initial_cache_size = len(_schema_cache) + + # Create a mock client + mock_client = MagicMock() + mock_client.chat.completions.create = AsyncMock() + + # Create the AsyncCompletions instance with our mock client + completions = AsyncCompletions(mock_client) + + # Simulate the issue by creating multiple models and making calls + models = [] + for i in range(10): + # Create a new dynamic model each time + new_model = create_model( + f"MathResponse{i}", + steps=(List[StepModel], Field()), + final_answer=(str, Field()), + ) + models.append(new_model) + + # Convert to response format and check if it's in the cache + type_to_response_format_param(new_model) + assert new_model in _schema_cache + + # Record cache size with all models referenced + cache_size_with_references = len(_schema_cache) + + # Let the models go out of scope and trigger garbage collection + models = None + gc.collect() + + # After garbage collection, the cache should be significantly reduced + cache_size_after_gc = len(_schema_cache) + assert cache_size_after_gc < cache_size_with_references + # The cache size should be close to the initial size (with some tolerance) + assert cache_size_after_gc < cache_size_with_references / 2 \ No newline at end of file From d5957256ef48060e44ed8b61b425d89726fee6ff Mon Sep 17 00:00:00 2001 From: mousberg Date: Fri, 28 Mar 2025 22:53:39 +0000 Subject: [PATCH 2/3] Fix lint issues, remove unittest class, add proper type annotations --- src/openai/lib/_parsing/_completions.py | 2 +- tests/lib/_parsing/test_memory_leak.py | 51 ------------------------- 2 files changed, 1 insertion(+), 52 deletions(-) diff --git a/src/openai/lib/_parsing/_completions.py b/src/openai/lib/_parsing/_completions.py index fc7e6e5ab8..2f4dd47ba4 100644 --- a/src/openai/lib/_parsing/_completions.py +++ b/src/openai/lib/_parsing/_completions.py @@ -262,7 +262,7 @@ def type_to_response_format_param( else: raise TypeError(f"Unsupported response_format type - {response_format}") - schema_param = { + schema_param: ResponseFormatParam = { "type": "json_schema", "json_schema": { "schema": to_strict_json_schema(json_schema_type), diff --git a/tests/lib/_parsing/test_memory_leak.py b/tests/lib/_parsing/test_memory_leak.py index 3e7dd46a22..4a83f4ec0b 100644 --- a/tests/lib/_parsing/test_memory_leak.py +++ b/tests/lib/_parsing/test_memory_leak.py @@ -1,56 +1,12 @@ -import unittest import gc -import sys -from unittest.mock import AsyncMock, patch, MagicMock from typing import List import pytest from pydantic import Field, create_model -from openai.resources.beta.chat.completions import AsyncCompletions from openai.lib._parsing import type_to_response_format_param from openai.lib._parsing._completions import _schema_cache -class TestMemoryLeak(unittest.TestCase): - def setUp(self): - # Clear the schema cache before each test - _schema_cache.clear() - - def test_schema_cache_with_models(self): - """Test if schema cache properly handles dynamic models and prevents memory leak""" - - StepModel = create_model( - "Step", - explanation=(str, Field()), - output=(str, Field()), - ) - - # Create several models and ensure they're cached properly - models = [] - for i in range(5): - model = create_model( - f"MathResponse{i}", - steps=(List[StepModel], Field()), - final_answer=(str, Field()), - ) - models.append(model) - - # Convert model to response format param - param = type_to_response_format_param(model) - - # Check if the model is in the cache - self.assertIn(model, _schema_cache) - - # Test that all models are in the cache - self.assertEqual(len(_schema_cache), 5) - - # Let the models go out of scope and trigger garbage collection - models = None - gc.collect() - - # After garbage collection, the cache should be empty or reduced - # since we're using weakref.WeakKeyDictionary - self.assertLess(len(_schema_cache), 5) @pytest.mark.asyncio async def test_async_completions_parse_memory(): @@ -65,13 +21,6 @@ async def test_async_completions_parse_memory(): _schema_cache.clear() initial_cache_size = len(_schema_cache) - # Create a mock client - mock_client = MagicMock() - mock_client.chat.completions.create = AsyncMock() - - # Create the AsyncCompletions instance with our mock client - completions = AsyncCompletions(mock_client) - # Simulate the issue by creating multiple models and making calls models = [] for i in range(10): From bcb75a8ed336cd9f8744c448308c4006fd45be49 Mon Sep 17 00:00:00 2001 From: mousberg Date: Thu, 23 Oct 2025 20:37:09 +0100 Subject: [PATCH 3/3] Fix lint errors and add type annotations - Remove unused initial_cache_size variable - Add type annotations for _schema_cache, test function return type - Use del instead of assignment to None for better garbage collection - Add type ignore comment for Pydantic dynamic model typing - All ruff checks pass --- src/openai/lib/_parsing/_completions.py | 2 +- tests/lib/_parsing/test_memory_leak.py | 14 +++++++------- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/src/openai/lib/_parsing/_completions.py b/src/openai/lib/_parsing/_completions.py index 6bc8c0b2e7..e946c7edf8 100644 --- a/src/openai/lib/_parsing/_completions.py +++ b/src/openai/lib/_parsing/_completions.py @@ -32,7 +32,7 @@ from ...types.chat.chat_completion_message_function_tool_call import Function # Cache to store weak references to schema objects -_schema_cache = weakref.WeakKeyDictionary() +_schema_cache: weakref.WeakKeyDictionary[type, ResponseFormatParam] = weakref.WeakKeyDictionary() ResponseFormatT = TypeVar( "ResponseFormatT", diff --git a/tests/lib/_parsing/test_memory_leak.py b/tests/lib/_parsing/test_memory_leak.py index 4a83f4ec0b..e60dcf8953 100644 --- a/tests/lib/_parsing/test_memory_leak.py +++ b/tests/lib/_parsing/test_memory_leak.py @@ -9,25 +9,25 @@ @pytest.mark.asyncio -async def test_async_completions_parse_memory(): +async def test_async_completions_parse_memory() -> None: """Test if AsyncCompletions.parse() doesn't leak memory with dynamic models""" + # Create a base step model StepModel = create_model( "Step", explanation=(str, Field()), output=(str, Field()), ) - # Clear the cache and record initial state + # Clear the cache before testing _schema_cache.clear() - initial_cache_size = len(_schema_cache) - + # Simulate the issue by creating multiple models and making calls - models = [] + models: list[type] = [] for i in range(10): # Create a new dynamic model each time new_model = create_model( f"MathResponse{i}", - steps=(List[StepModel], Field()), + steps=(List[StepModel], Field()), # type: ignore[valid-type] final_answer=(str, Field()), ) models.append(new_model) @@ -40,7 +40,7 @@ async def test_async_completions_parse_memory(): cache_size_with_references = len(_schema_cache) # Let the models go out of scope and trigger garbage collection - models = None + del models gc.collect() # After garbage collection, the cache should be significantly reduced