Skip to content

Commit 230f225

Browse files
committed
fixes #420 Where if the key of a dictionary contains the characters used in the path the path is actually corrupted.
1 parent a9b7286 commit 230f225

7 files changed

+58
-7
lines changed

deepdiff/delta.py

+1
Original file line numberDiff line numberDiff line change
@@ -623,6 +623,7 @@ def to_flat_dicts(self, include_action_in_path=False, report_type_changes=True):
623623
include_action_in_path : Boolean, default=False
624624
When False, we translate DeepDiff's paths like root[3].attribute1 into a [3, 'attribute1'].
625625
When True, we include the action to retrieve the item in the path: [(3, 'GET'), ('attribute1', 'GETATTR')]
626+
Note that the "action" here is the different than the action reported by to_flat_dicts. The action here is just about the "path" output.
626627
627628
report_type_changes : Boolean, default=True
628629
If False, we don't report the type change. Instead we report the value change.

deepdiff/model.py

+15-1
Original file line numberDiff line numberDiff line change
@@ -874,7 +874,21 @@ def stringify_param(self, force=None):
874874
"""
875875
param = self.param
876876
if isinstance(param, strings):
877-
result = param if self.quote_str is None else self.quote_str.format(param)
877+
has_quote = "'" in param
878+
has_double_quote = '"' in param
879+
if has_quote and has_double_quote:
880+
new_param = []
881+
for char in param:
882+
if char in {'"', "'"}:
883+
new_param.append('\\')
884+
new_param.append(char)
885+
param = ''.join(new_param)
886+
elif has_quote:
887+
result = f'"{param}"'
888+
elif has_double_quote:
889+
result = f"'{param}'"
890+
else:
891+
result = param if self.quote_str is None else self.quote_str.format(param)
878892
elif isinstance(param, tuple): # Currently only for numpy ndarrays
879893
result = ']['.join(map(repr, param))
880894
else:

deepdiff/path.py

+10-4
Original file line numberDiff line numberDiff line change
@@ -53,15 +53,21 @@ def _path_to_elements(path, root_element=DEFAULT_FIRST_ELEMENT):
5353
path = path[4:] # removing "root from the beginning"
5454
brackets = []
5555
inside_quotes = False
56+
quote_used = ''
5657
for char in path:
5758
if prev_char == '\\':
5859
elem += char
5960
elif char in {'"', "'"}:
6061
elem += char
61-
inside_quotes = not inside_quotes
62-
if not inside_quotes:
63-
_add_to_elements(elements, elem, inside)
64-
elem = ''
62+
# If we are inside and the quote is not what we expected, the quote is not closing
63+
if not(inside_quotes and quote_used != char):
64+
inside_quotes = not inside_quotes
65+
if inside_quotes:
66+
quote_used = char
67+
else:
68+
_add_to_elements(elements, elem, inside)
69+
elem = ''
70+
quote_used = ''
6571
elif inside_quotes:
6672
elem += char
6773
elif char == '[':

tests/test_delta.py

+7
Original file line numberDiff line numberDiff line change
@@ -681,6 +681,13 @@ def test_delta_dict_items_added_retain_order(self):
681681
'to_delta_kwargs': {'directed': True},
682682
'expected_delta_dict': {'iterable_item_removed': {'root[4]': 4}}
683683
},
684+
'delta_case20_quotes_in_path': {
685+
't1': {"a']['b']['c": 1},
686+
't2': {"a']['b']['c": 2},
687+
'deepdiff_kwargs': {},
688+
'to_delta_kwargs': {'directed': True},
689+
'expected_delta_dict': {'values_changed': {'root["a\'][\'b\'][\'c"]': {'new_value': 2}}}
690+
},
684691
}
685692

686693

tests/test_diff_text.py

+11
Original file line numberDiff line numberDiff line change
@@ -297,6 +297,17 @@ def test_string_difference_ignore_case(self):
297297
result = {}
298298
assert result == ddiff
299299

300+
def test_diff_quote_in_string(self):
301+
t1 = {
302+
"a']['b']['c": 1
303+
}
304+
t2 = {
305+
"a']['b']['c": 2
306+
}
307+
diff = DeepDiff(t1, t2)
308+
expected = {'values_changed': {'''root["a']['b']['c"]''': {'new_value': 2, 'old_value': 1}}}
309+
assert expected == diff
310+
300311
def test_bytes(self):
301312
t1 = {
302313
1: 1,

tests/test_path.py

+12-2
Original file line numberDiff line numberDiff line change
@@ -24,10 +24,20 @@ def test_path_to_elements(path, expected):
2424
5),
2525
({1: [{'2': 'b'}, 3], 2: {4, 5}},
2626
"root[1][0]['2']",
27-
'b'),
27+
'b'
28+
),
2829
({'test [a]': 'b'},
2930
"root['test [a]']",
30-
'b'),
31+
'b'
32+
),
33+
({"a']['b']['c": 1},
34+
"""root["a\\'][\\'b\\'][\\'c"]""",
35+
1
36+
),
37+
({"a']['b']['c": 1},
38+
"""root["a']['b']['c"]""",
39+
1
40+
),
3141
])
3242
def test_get_item(obj, path, expected):
3343
result = extract(obj, path)

tests/test_serialization.py

+2
Original file line numberDiff line numberDiff line change
@@ -321,6 +321,8 @@ def test_pretty_form_method(self, expected, verbose_level):
321321
(3, {'10': Decimal(2017)}, None),
322322
(4, Decimal(2017.1), None),
323323
(5, {1, 2, 10}, set),
324+
(6, datetime.datetime(2023, 10, 11), datetime.datetime.fromisoformat),
325+
(7, datetime.datetime.utcnow(), datetime.datetime.fromisoformat),
324326
])
325327
def test_json_dumps_and_loads(self, test_num, value, func_to_convert_back):
326328
serialized = json_dumps(value)

0 commit comments

Comments
 (0)