Dal momento che sembra che nessuno ne abbia parlato deepdiff, lo aggiungerò qui per completezza. Lo trovo molto conveniente per ottenere diff di oggetti (nidificati) in generale:
Installazione
pip install deepdiff
Codice di esempio
import deepdiff
import json
dict_1 = {
"a": 1,
"nested": {
"b": 1,
}
}
dict_2 = {
"a": 2,
"nested": {
"b": 2,
}
}
diff = deepdiff.DeepDiff(dict_1, dict_2)
print(json.dumps(diff, indent=4))
Produzione
{
"values_changed": {
"root['a']": {
"new_value": 2,
"old_value": 1
},
"root['nested']['b']": {
"new_value": 2,
"old_value": 1
}
}
}
Nota sulla stampa graziosa del risultato per l'ispezione: il codice sopra riportato funziona se entrambi i dadi hanno le stesse chiavi di attributo (con valori di attributo possibilmente diversi come nell'esempio). Tuttavia, se un "extra"attributo è presente è una delle dicts, json.dumps()fallisce con
TypeError: Object of type PrettyOrderedSet is not JSON serializable
Soluzione: utilizzare diff.to_json()e json.loads()/ json.dumps()per stampare piuttosto:
import deepdiff
import json
dict_1 = {
"a": 1,
"nested": {
"b": 1,
},
"extra": 3
}
dict_2 = {
"a": 2,
"nested": {
"b": 2,
}
}
diff = deepdiff.DeepDiff(dict_1, dict_2)
print(json.dumps(json.loads(diff.to_json()), indent=4))
Produzione:
{
"dictionary_item_removed": [
"root['extra']"
],
"values_changed": {
"root['a']": {
"new_value": 2,
"old_value": 1
},
"root['nested']['b']": {
"new_value": 2,
"old_value": 1
}
}
}
Alternativa: uso pprint, risulta in una formattazione diversa:
import pprint
# same code as above
pprint.pprint(diff, indent=4)
Produzione:
{ 'dictionary_item_removed': [root['extra']],
'values_changed': { "root['a']": { 'new_value': 2,
'old_value': 1},
"root['nested']['b']": { 'new_value': 2,
'old_value': 1}}}
x == ydovrebbe essere vero in base alle stackoverflow.com/a/5635309/186202