Skip to content

Commit 27d850e

Browse files
committed
fix(logger): no longer replaces object duplicates with "CIRCULAR"
1 parent e1110eb commit 27d850e

File tree

2 files changed

+163
-5
lines changed

2 files changed

+163
-5
lines changed

src/firebase_functions/logger.py

+16-5
Original file line numberDiff line numberDiff line change
@@ -75,21 +75,32 @@ def _remove_circular(obj: _typing.Any,
7575
if refs is None:
7676
refs = set()
7777

78+
# Check if the object is already in the current recursion stack
7879
if id(obj) in refs:
7980
return "[CIRCULAR]"
8081

82+
# For non-primitive objects, add the current object's id to the recursion stack
8183
if not isinstance(obj, (str, int, float, bool, type(None))):
8284
refs.add(id(obj))
8385

86+
# Recursively process the object based on its type
87+
result: _typing.Any
8488
if isinstance(obj, dict):
85-
return {key: _remove_circular(value, refs) for key, value in obj.items()}
89+
result = {
90+
key: _remove_circular(value, refs) for key, value in obj.items()
91+
}
8692
elif isinstance(obj, list):
87-
return [_remove_circular(value, refs) for _, value in enumerate(obj)]
93+
result = [_remove_circular(item, refs) for item in obj]
8894
elif isinstance(obj, tuple):
89-
return tuple(
90-
_remove_circular(value, refs) for _, value in enumerate(obj))
95+
result = tuple(_remove_circular(item, refs) for item in obj)
9196
else:
92-
return obj
97+
result = obj
98+
99+
# Remove the object's id from the recursion stack after processing
100+
if not isinstance(obj, (str, int, float, bool, type(None))):
101+
refs.remove(id(obj))
102+
103+
return result
93104

94105

95106
def _get_write_file(severity: LogSeverity) -> _typing.TextIO:

tests/test_logger.py

+147
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,13 @@
1+
# mypy: ignore-errors
2+
13
"""
24
Logger module tests.
35
"""
46

57
import pytest
68
import json
79
from firebase_functions import logger
10+
import typing as _typing
811

912

1013
class TestLogger:
@@ -79,3 +82,147 @@ def test_message_should_be_space_separated(
7982
raw_log_output = capsys.readouterr().out
8083
log_output = json.loads(raw_log_output)
8184
assert log_output["message"] == expected_message
85+
86+
def test_remove_circular_references(self,
87+
capsys: pytest.CaptureFixture[str]):
88+
# Create an object with a circular reference.
89+
circ: _typing.Any = {"b": "foo"}
90+
circ["circ"] = circ
91+
92+
entry = {
93+
"severity": "ERROR",
94+
"message": "testing circular",
95+
"circ": circ,
96+
} # i
97+
logger.write(entry)
98+
raw_log_output = capsys.readouterr().err
99+
log_output = json.loads(raw_log_output)
100+
101+
expected = {
102+
"severity": "ERROR",
103+
"message": "testing circular",
104+
"circ": {
105+
"b": "foo",
106+
"circ": "[CIRCULAR]"
107+
},
108+
}
109+
assert log_output == expected
110+
111+
def test_remove_circular_references_in_arrays(
112+
self, capsys: pytest.CaptureFixture[str]):
113+
# Create an object with a circular reference inside an array.
114+
circ: _typing.Any = {"b": "foo"}
115+
circ["circ"] = [circ]
116+
117+
entry = {
118+
"severity": "ERROR",
119+
"message": "testing circular",
120+
"circ": circ,
121+
}
122+
logger.write(entry)
123+
raw_log_output = capsys.readouterr().err
124+
log_output = json.loads(raw_log_output)
125+
126+
expected = {
127+
"severity": "ERROR",
128+
"message": "testing circular",
129+
"circ": {
130+
"b": "foo",
131+
"circ": ["[CIRCULAR]"]
132+
},
133+
}
134+
assert log_output == expected
135+
136+
def test_no_false_circular_for_duplicates(
137+
self, capsys: pytest.CaptureFixture[str]):
138+
# Ensure that duplicate objects (used in multiple keys) are not marked as circular.
139+
obj = {"a": "foo"}
140+
entry = {
141+
"severity": "ERROR",
142+
"message": "testing circular",
143+
"a": obj,
144+
"b": obj,
145+
}
146+
logger.write(entry)
147+
raw_log_output = capsys.readouterr().err
148+
log_output = json.loads(raw_log_output)
149+
150+
expected = {
151+
"severity": "ERROR",
152+
"message": "testing circular",
153+
"a": {
154+
"a": "foo"
155+
},
156+
"b": {
157+
"a": "foo"
158+
},
159+
}
160+
assert log_output == expected
161+
162+
def test_no_false_circular_in_array_duplicates(
163+
self, capsys: pytest.CaptureFixture[str]):
164+
# Ensure that duplicate objects in arrays are not falsely detected as circular.
165+
obj = {"a": "foo"}
166+
arr = [
167+
{
168+
"a": obj,
169+
"b": obj
170+
},
171+
{
172+
"a": obj,
173+
"b": obj
174+
},
175+
]
176+
entry = {
177+
"severity": "ERROR",
178+
"message": "testing circular",
179+
"a": arr,
180+
"b": arr,
181+
}
182+
logger.write(entry)
183+
raw_log_output = capsys.readouterr().err
184+
log_output = json.loads(raw_log_output)
185+
186+
expected = {
187+
"severity":
188+
"ERROR",
189+
"message":
190+
"testing circular",
191+
"a": [
192+
{
193+
"a": {
194+
"a": "foo"
195+
},
196+
"b": {
197+
"a": "foo"
198+
}
199+
},
200+
{
201+
"a": {
202+
"a": "foo"
203+
},
204+
"b": {
205+
"a": "foo"
206+
}
207+
},
208+
],
209+
"b": [
210+
{
211+
"a": {
212+
"a": "foo"
213+
},
214+
"b": {
215+
"a": "foo"
216+
}
217+
},
218+
{
219+
"a": {
220+
"a": "foo"
221+
},
222+
"b": {
223+
"a": "foo"
224+
}
225+
},
226+
],
227+
}
228+
assert log_output == expected

0 commit comments

Comments
 (0)