Skip to content

Commit 850b468

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

File tree

2 files changed

+161
-5
lines changed

2 files changed

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

+145
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
import pytest
66
import json
77
from firebase_functions import logger
8+
from typing import Any
89

910

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

0 commit comments

Comments
 (0)