Skip to content

Commit d934708

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

File tree

2 files changed

+161
-6
lines changed

2 files changed

+161
-6
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

+145-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
# mypy: ignore-errors
12
"""
23
Logger module tests.
34
"""
@@ -6,7 +7,6 @@
67
import json
78
from firebase_functions import logger
89

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

0 commit comments

Comments
 (0)