mirror of
https://github.com/cwinfo/powerdns-admin.git
synced 2025-01-09 20:05:39 +00:00
Merge pull request #1423 from corubba/feature/history-diff
Diff-ify changelog view for zone changes
This commit is contained in:
commit
840076dae3
@ -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
|
||||||
|
@ -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: "+";
|
||||||
|
}
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
<tbody>
|
||||||
|
{% for changes in hist_rec_entry.changeSet %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
<td style="word-break: break-all">
|
||||||
<table>
|
{% if changes[2] == "unchanged" or
|
||||||
<tbody>
|
(changes[2] == "edit" and changes[0]['disabled'] == changes[1]['disabled']) %}
|
||||||
{% for changes in hist_rec_entry.changeSet %}
|
<div class="diff diff-unchanged">{{
|
||||||
<tr>
|
"Disabled" if changes[1]['disabled'] else "Activated"
|
||||||
{% if changes[2] == "unchanged" %}
|
}}</div>
|
||||||
<td>{{ "Activated" if changes[0]['disabled'] ==
|
{% else %}
|
||||||
False else
|
{% if changes[2] in ["deletion", "edit"] %}
|
||||||
"Disabled"}} </td>
|
<div class="diff diff-deletion">{{
|
||||||
{% elif changes[2] == "addition" %}
|
"Disabled" if changes[0]['disabled'] else "Activated"
|
||||||
<td>
|
}}</div>
|
||||||
<span style="background-color: lightgreen">
|
{% endif %}
|
||||||
{{ "Activated" if changes[1]['disabled'] ==
|
{% if changes[2] in ["addition", "edit"] %}
|
||||||
False else
|
<div class="diff diff-addition">{{
|
||||||
"Disabled"}}
|
"Disabled" if changes[1]['disabled'] else "Activated"
|
||||||
</span>
|
}}</div>
|
||||||
</td>
|
{% endif %}
|
||||||
{% elif changes[2] == "status" %}
|
{% endif %}
|
||||||
<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>
|
|
||||||
<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 %}
|
|
||||||
</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 %}
|
|
||||||
</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 %}
|
||||||
|
Loading…
Reference in New Issue
Block a user