Diff-ify changelog view for zone changes

Improve and document the diff-computation and presentation, so you can
easier see what changed.
This commit is contained in:
corubba 2023-03-03 13:22:29 +01:00
parent 077bbb813c
commit 8a40d21ea4
4 changed files with 186 additions and 160 deletions

View File

@ -34,61 +34,77 @@ admin_bp = Blueprint('admin',
template_folder='templates', template_folder='templates',
url_prefix='/admin') url_prefix='/admin')
"""
changeSet is a list of tuples, in the following format
(old_state, new_state, change_type)
old_state: dictionary with "disabled" and "content" keys. {"disabled" : False, "content" : "1.1.1.1" }
new_state: similarly
change_type: "addition" or "deletion" or "status" for status change or "unchanged" for no change
Note: A change in "content", is considered a deletion and recreation of the same record,
holding the new content value.
"""
def get_record_changes(del_rrset, add_rrset): def get_record_changes(del_rrset, add_rrset):
changeSet = [] """Use the given deleted and added RRset to build a list of record changes.
delSet = del_rrset['records'] if 'records' in del_rrset else []
addSet = add_rrset['records'] if 'records' in add_rrset else [] Args:
for d in delSet: # get the deletions and status changes del_rrset: The RRset with changetype DELETE, or None
exists = False add_rrset: The RRset with changetype REPLACE, or None
for a in addSet:
if d['content'] == a['content']: Returns:
exists = True A list of tuples in the format `(old_state, new_state, change_type)`. `old_state` and
if d['disabled'] != a['disabled']: `new_state` are dictionaries with the keys "disabled", "content" and "comment".
changeSet.append(({"disabled": d['disabled'], "content": d['content']}, `change_type` can be "addition", "deletion", "edit" or "unchanged". When it's "addition"
{"disabled": a['disabled'], "content": a['content']}, then `old_state` is None, when it's "deletion" then `new_state` is None.
"status")) """
def get_records(rrset):
"""For the given RRset return a combined list of records and comments."""
if not rrset or 'records' not in rrset:
return []
records = [dict(record) for record in rrset['records']]
for i, record in enumerate(records):
if 'comments' in rrset and len(rrset['comments']) > i:
record['comment'] = rrset['comments'][i].get('content', None)
else:
record['comment'] = None
return records
def record_is_unchanged(old, new):
"""Returns True if the old record is not different from the new one."""
if old['content'] != new['content']:
raise ValueError("Can't compare records with different content")
# check everything except the content
return old['disabled'] == new['disabled'] and old['comment'] == new['comment']
def to_state(record):
"""For the given record, return the state dict."""
return {
"disabled": record['disabled'],
"content": record['content'],
"comment": record.get('comment', ''),
}
add_records = get_records(add_rrset)
del_records = get_records(del_rrset)
changeset = []
for add_record in add_records:
for del_record in list(del_records):
if add_record['content'] == del_record['content']:
# either edited or unchanged
if record_is_unchanged(del_record, add_record):
# unchanged
changeset.append((to_state(del_record), to_state(add_record), "unchanged"))
else:
# edited
changeset.append((to_state(del_record), to_state(add_record), "edit"))
del_records.remove(del_record)
break break
else: # not mis-indented, else block for the del_records for loop
# addition
changeset.append((None, to_state(add_record), "addition"))
if not exists: # deletion # Because the first loop removed edit/unchanged records from the del_records list,
changeSet.append(({"disabled": d['disabled'], "content": d['content']}, # it now only contains real deletions.
None, for del_record in del_records:
"deletion")) changeset.append((to_state(del_record), None, "deletion"))
for a in addSet: # get the additions # Sort them by the old content. For Additions the new state will be used.
exists = False changeset.sort(key=lambda change: change[0]['content'] if change[0] else change[1]['content'])
for d in delSet:
if d['content'] == a['content']:
exists = True
# already checked for status change
break
if not exists:
changeSet.append((None, {"disabled": a['disabled'], "content": a['content']}, "addition"))
continue
for a in addSet: # get the unchanged return changeset
exists = False
for c in changeSet:
if c[1] != None and c[1]["content"] == a['content']:
exists = True
break
if not exists:
changeSet.append(({"disabled": a['disabled'], "content": a['content']},
{"disabled": a['disabled'], "content": a['content']}, "unchanged"))
return changeSet
# out_changes is a list of HistoryRecordEntry objects in which we will append the new changes # out_changes is a list of HistoryRecordEntry objects in which we will append the new changes
@ -118,7 +134,7 @@ def extract_changelogs_from_a_history_entry(out_changes, history_entry, change_n
if change_num not in out_changes: if change_num not in out_changes:
out_changes[change_num] = [] out_changes[change_num] = []
out_changes[change_num].append( out_changes[change_num].append(
HistoryRecordEntry(history_entry, [], add_rrset, "+")) # (add_rrset, del_rrset, change_type) HistoryRecordEntry(history_entry, {}, add_rrset, "+")) # (add_rrset, del_rrset, change_type)
for del_rrset in del_rrsets: for del_rrset in del_rrsets:
exists = False exists = False
for add_rrset in add_rrsets: for add_rrset in add_rrsets:
@ -128,7 +144,13 @@ def extract_changelogs_from_a_history_entry(out_changes, history_entry, change_n
if not exists: # this is a deletion if not exists: # this is a deletion
if change_num not in out_changes: if change_num not in out_changes:
out_changes[change_num] = [] out_changes[change_num] = []
out_changes[change_num].append(HistoryRecordEntry(history_entry, del_rrset, [], "-")) out_changes[change_num].append(HistoryRecordEntry(history_entry, del_rrset, {}, "-"))
# Sort them by the record name
if change_num in out_changes:
out_changes[change_num].sort(key=lambda change:
change.del_rrset['name'] if change.del_rrset else change.add_rrset['name']
)
# only used for changelog per record # only used for changelog per record
if record_name != None and record_type != None: # then get only the records with the specific (record_name, record_type) tuple if record_name != None and record_type != None: # then get only the records with the specific (record_name, record_type) tuple

View File

@ -57,3 +57,26 @@ table.records thead th:last-of-type { width: 50px; }
div.records > div.dataTables_wrapper > div.row:first-of-type { margin: 0 0.5em 0 0.5em; } div.records > div.dataTables_wrapper > div.row:first-of-type { margin: 0 0.5em 0 0.5em; }
div.records > div.dataTables_wrapper > div.row:last-of-type { margin: 0.4em 0.5em 0.4em 0.5em; } div.records > div.dataTables_wrapper > div.row:last-of-type { margin: 0.4em 0.5em 0.4em 0.5em; }
div.records > div.dataTables_wrapper table.dataTable { margin: 0 !important; } div.records > div.dataTables_wrapper table.dataTable { margin: 0 !important; }
.diff {
font-family: monospace;
padding: 0 0.2em;
}
.diff::before {
content: "\00a0";
padding-right: 0.1em;
}
.diff-deletion {
background-color: lightcoral;
}
.diff-deletion::before {
content: "-";
}
.diff-addition {
background-color: lightgreen;
}
.diff-addition::before {
content: "+";
}

View File

@ -537,7 +537,7 @@
</div> </div>
<!-- History Details Box --> <!-- History Details Box -->
<div class="modal fade" id="modal_history_info"> <div class="modal fade" id="modal_history_info">
<div class="modal-dialog"> <div class="modal-dialog modal-lg">
<div class="modal-content"> <div class="modal-content">
<div class="modal-header"> <div class="modal-header">
<h4 class="modal-title">History Details</h4> <h4 class="modal-title">History Details</h4>

View File

@ -4,129 +4,110 @@
<table id="tbl_records" class="table table-bordered"> <table id="tbl_records" class="table table-bordered">
<thead> <thead>
<tr> <tr>
<th colspan="3"> <th scope="col" colspan="3">
{% if hist_rec_entry.change_type == "+" %} {% if hist_rec_entry.change_type == '-' %}
<span <span class="diff diff-deletion">{{
style="background-color: lightgreen">{{hist_rec_entry.add_rrset['name']}} hist_rec_entry.del_rrset['name'] }} {{ hist_rec_entry.del_rrset['type']
{{hist_rec_entry.add_rrset['type']}}</span> }}</span>
{% elif hist_rec_entry.change_type == "-" %} {% elif hist_rec_entry.change_type == '+' %}
<s <span class="diff diff-addition">{{
style="text-decoration-color: rgba(194, 10,10, 0.6); text-decoration-thickness: 2px;"> hist_rec_entry.add_rrset['name'] }} {{ hist_rec_entry.add_rrset['type']
{{hist_rec_entry.del_rrset['name']}} }}</span>
{{hist_rec_entry.del_rrset['type']}}
</s>
{% else %} {% else %}
{{hist_rec_entry.add_rrset['name']}} <span class="diff diff-unchanged">{{
{{hist_rec_entry.add_rrset['type']}} hist_rec_entry.add_rrset['name'] }} {{ hist_rec_entry.add_rrset['type']
}}</span>
{% endif %} {% endif %}
, TTL: , TTL
{% if "ttl" in hist_rec_entry.changed_fields %} {% if not 'ttl' in hist_rec_entry.changed_fields %}
<s <span class="diff diff-unchanged">{{
style="text-decoration-color: rgba(194, 10,10, 0.6); text-decoration-thickness: 2px;"> hist_rec_entry.add_rrset['ttl']
{{hist_rec_entry.del_rrset['ttl']}}</s> }}</span>
<span
style="background-color: lightgreen">{{hist_rec_entry.add_rrset['ttl']}}</span>
{% else %} {% else %}
{{hist_rec_entry.add_rrset['ttl']}} {% if hist_rec_entry.change_type in ['-', '*'] %}
<span class="diff diff-deletion">{{
hist_rec_entry.del_rrset['ttl']
}}</span>
{% endif %}
{% if hist_rec_entry.change_type in ['+', '*'] %}
<span class="diff diff-addition">{{
hist_rec_entry.add_rrset['ttl']
}}</span>
{% endif %}
{% endif %} {% endif %}
</th> </th>
</tr> </tr>
<tr> <tr>
<th scope="col" style="width: 150px;">Status</th>
<th style="width: 150px;">Status</th> <th scope="col" style="width: 400px;">Data</th>
<th style="width: 400px;">Data</th> <th scope="col" style="width: 400px;">Comment</th>
<th style="width: 400px;">Comment</th>
</tr> </tr>
</thead> </thead>
<tbody>
<tr>
<td>
<table>
<tbody> <tbody>
{% for changes in hist_rec_entry.changeSet %} {% for changes in hist_rec_entry.changeSet %}
<tr> <tr>
{% if changes[2] == "unchanged" %} <td style="word-break: break-all">
<td>{{ "Activated" if changes[0]['disabled'] == {% if changes[2] == "unchanged" or
False else (changes[2] == "edit" and changes[0]['disabled'] == changes[1]['disabled']) %}
"Disabled"}} </td> <div class="diff diff-unchanged">{{
{% elif changes[2] == "addition" %} "Disabled" if changes[1]['disabled'] else "Activated"
<td> }}</div>
<span style="background-color: lightgreen"> {% else %}
{{ "Activated" if changes[1]['disabled'] == {% if changes[2] in ["deletion", "edit"] %}
False else <div class="diff diff-deletion">{{
"Disabled"}} "Disabled" if changes[0]['disabled'] else "Activated"
</span> }}</div>
</td> {% endif %}
{% elif changes[2] == "status" %} {% if changes[2] in ["addition", "edit"] %}
<td> <div class="diff diff-addition">{{
<s "Disabled" if changes[1]['disabled'] else "Activated"
style="text-decoration-color: rgba(194, 10,10, 0.6); text-decoration-thickness: 2px;"> }}</div>
{{ "Activated" if changes[0]['disabled'] ==
False else
"Disabled"}}</s>
<span style="background-color: lightgreen">{{
"Activated" if changes[1]['disabled'] ==
False else
"Disabled"}}</span>
</td>
{% elif changes[2] == "deletion" %}
<td>
<s
style="text-decoration-color: rgba(194, 10,10, 0.6); text-decoration-thickness: 2px;">
{{ "Activated" if changes[0]['disabled'] ==
False else
"Disabled"}}</s>
</td>
{% endif %} {% endif %}
</tr>
{% endfor %}
</tbody>
</table>
</td>
<td>
<table>
<tbody>
{% for changes in hist_rec_entry.changeSet %}
<tr>
{% if changes[2] == "unchanged" %}
<td style="word-break: break-all">
{{changes[0]['content']}}
</td>
{% elif changes[2] == "addition" %}
<td style="word-break: break-all">
<span style="background-color: lightgreen">
{{changes[1]['content']}}
</span>
</td>
{% elif changes[2] == "deletion" %}
<td style="word-break: break-all">
<s
style="text-decoration-color: rgba(194, 10, 10, 0.6); text-decoration-thickness: 2px;">
{{changes[0]['content']}}
</s>
</td>
{% elif changes[2] == "status" %}
<td style="word-break: break-all">
{{changes[0]['content']}}
</td>
{% endif %} {% endif %}
</tr>
{% endfor %}
</tbody>
</table>
</td> </td>
<td style="word-break: break-all"> <td style="word-break: break-all">
{% for comments in hist_rec_entry.add_rrset['comments'] %} {% if changes[2] == "unchanged" or
{{comments['content'] }} (changes[2] == "edit" and changes[0]['content'] == changes[1]['content']) %}
<br/> <div class="diff diff-unchanged">{{
{% endfor %} changes[1]['content']
}}</div>
{% else %}
{% if changes[2] in ["deletion", "edit"] %}
<div class="diff diff-deletion">{{
changes[0]['content']
}}</div>
{% endif %}
{% if changes[2] in ["addition", "edit"] %}
<div class="diff diff-addition">{{
changes[1]['content']
}}</div>
{% endif %}
{% endif %}
</td>
<td style="word-break: break-all">
{% if changes[2] == "unchanged" or
(changes[2] == "edit" and changes[0]['comment'] == changes[1]['comment']) %}
<div class="diff diff-unchanged">{{
changes[1]['comment']
}}</div>
{% else %}
{% if changes[2] in ["deletion", "edit"] and changes[0]['comment'] %}
<div class="diff diff-deletion">{{
changes[0]['comment']
}}</div>
{% endif %}
{% if changes[2] in ["addition", "edit"] and changes[1]['comment'] %}
<div class="diff diff-addition">{{
changes[1]['comment']
}}</div>
{% endif %}
{% endif %}
</td> </td>
</tr> </tr>
{% endfor %}
</tbody> </tbody>
</table> </table>
{% endfor %} {% endfor %}