mirror of
https://github.com/jazzband/django-auditlog.git
synced 2026-03-16 22:20:26 +00:00
Compare commits
7 commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0e3a2ec1a7 | ||
|
|
dfd5b79d2d | ||
|
|
4154560de3 | ||
|
|
3f255a02d9 | ||
|
|
6d170da5fc | ||
|
|
198c060c3b | ||
|
|
ede4d10164 |
6 changed files with 64 additions and 18 deletions
|
|
@ -1,7 +1,7 @@
|
||||||
---
|
---
|
||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/psf/black-pre-commit-mirror
|
- repo: https://github.com/psf/black-pre-commit-mirror
|
||||||
rev: 25.11.0
|
rev: 26.3.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: black
|
- id: black
|
||||||
language_version: python3.10
|
language_version: python3.10
|
||||||
|
|
@ -14,7 +14,7 @@ repos:
|
||||||
- id: flake8
|
- id: flake8
|
||||||
args: ["--max-line-length", "110"]
|
args: ["--max-line-length", "110"]
|
||||||
- repo: https://github.com/PyCQA/isort
|
- repo: https://github.com/PyCQA/isort
|
||||||
rev: 7.0.0
|
rev: 8.0.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: isort
|
- id: isort
|
||||||
- repo: https://github.com/asottile/pyupgrade
|
- repo: https://github.com/asottile/pyupgrade
|
||||||
|
|
@ -23,7 +23,7 @@ repos:
|
||||||
- id: pyupgrade
|
- id: pyupgrade
|
||||||
args: [--py310-plus]
|
args: [--py310-plus]
|
||||||
- repo: https://github.com/adamchainz/django-upgrade
|
- repo: https://github.com/adamchainz/django-upgrade
|
||||||
rev: 1.29.1
|
rev: 1.30.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: django-upgrade
|
- id: django-upgrade
|
||||||
args: [--target-version, "4.2"]
|
args: [--target-version, "4.2"]
|
||||||
|
|
|
||||||
|
|
@ -2,6 +2,10 @@
|
||||||
|
|
||||||
## Next Release
|
## Next Release
|
||||||
|
|
||||||
|
#### Fixes
|
||||||
|
|
||||||
|
- `KeyError` when calling `changes_str` on a log entry that tracks many-to-many field changes ([#798](https://github.com/jazzband/django-auditlog/pull/798))
|
||||||
|
|
||||||
## 3.4.1 (2025-12-13)
|
## 3.4.1 (2025-12-13)
|
||||||
|
|
||||||
#### Fixes
|
#### Fixes
|
||||||
|
|
|
||||||
|
|
@ -126,15 +126,13 @@ class Command(BaseCommand):
|
||||||
|
|
||||||
def postgres():
|
def postgres():
|
||||||
with connection.cursor() as cursor:
|
with connection.cursor() as cursor:
|
||||||
cursor.execute(
|
cursor.execute(f"""
|
||||||
f"""
|
|
||||||
UPDATE {LogEntry._meta.db_table}
|
UPDATE {LogEntry._meta.db_table}
|
||||||
SET changes="changes_text"::jsonb
|
SET changes="changes_text"::jsonb
|
||||||
WHERE changes_text IS NOT NULL
|
WHERE changes_text IS NOT NULL
|
||||||
AND changes_text <> ''
|
AND changes_text <> ''
|
||||||
AND changes IS NULL
|
AND changes IS NULL
|
||||||
"""
|
""")
|
||||||
)
|
|
||||||
return cursor.cursor.rowcount
|
return cursor.cursor.rowcount
|
||||||
|
|
||||||
if database == "postgres":
|
if database == "postgres":
|
||||||
|
|
|
||||||
|
|
@ -427,21 +427,29 @@ class AbstractLogEntry(models.Model):
|
||||||
not satisfying, please use :py:func:`LogEntry.changes_dict` and format the string yourself.
|
not satisfying, please use :py:func:`LogEntry.changes_dict` and format the string yourself.
|
||||||
|
|
||||||
:param colon: The string to place between the field name and the values.
|
:param colon: The string to place between the field name and the values.
|
||||||
:param arrow: The string to place between each old and new value.
|
:param arrow: The string to place between each old and new value (non-m2m field changes only).
|
||||||
:param separator: The string to place between each field.
|
:param separator: The string to place between each field.
|
||||||
:return: A readable string of the changes in this log entry.
|
:return: A readable string of the changes in this log entry.
|
||||||
"""
|
"""
|
||||||
substrings = []
|
substrings = []
|
||||||
|
|
||||||
for field, values in self.changes_dict.items():
|
for field, value in sorted(self.changes_dict.items()):
|
||||||
substring = "{field_name:s}{colon:s}{old:s}{arrow:s}{new:s}".format(
|
if isinstance(value, (list, tuple)) and len(value) == 2:
|
||||||
field_name=field,
|
# handle regular field change
|
||||||
colon=colon,
|
substring = "{field_name:s}{colon:s}{old:s}{arrow:s}{new:s}".format(
|
||||||
old=values[0],
|
field_name=field,
|
||||||
arrow=arrow,
|
colon=colon,
|
||||||
new=values[1],
|
old=value[0],
|
||||||
)
|
arrow=arrow,
|
||||||
substrings.append(substring)
|
new=value[1],
|
||||||
|
)
|
||||||
|
substrings.append(substring)
|
||||||
|
elif isinstance(value, dict) and value.get("type") == "m2m":
|
||||||
|
# handle m2m change
|
||||||
|
substring = (
|
||||||
|
f"{field}{colon}{value['operation']} {sorted(value['objects'])}"
|
||||||
|
)
|
||||||
|
substrings.append(substring)
|
||||||
|
|
||||||
return separator.join(substrings)
|
return separator.join(substrings)
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -131,6 +131,11 @@ class SimpleModelTest(TestCase):
|
||||||
{"boolean": ["False", "True"]},
|
{"boolean": ["False", "True"]},
|
||||||
msg="The change is correctly logged",
|
msg="The change is correctly logged",
|
||||||
)
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
history.changes_str,
|
||||||
|
"boolean: False → True",
|
||||||
|
msg="Changes string is correct",
|
||||||
|
)
|
||||||
|
|
||||||
def test_update_specific_field_supplied_via_save_method(self):
|
def test_update_specific_field_supplied_via_save_method(self):
|
||||||
obj = self.obj
|
obj = self.obj
|
||||||
|
|
@ -149,6 +154,11 @@ class SimpleModelTest(TestCase):
|
||||||
"when using the `update_fields`."
|
"when using the `update_fields`."
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
obj.history.get(action=LogEntry.Action.UPDATE).changes_str,
|
||||||
|
"boolean: False → True",
|
||||||
|
msg="Changes string is correct",
|
||||||
|
)
|
||||||
|
|
||||||
def test_django_update_fields_edge_cases(self):
|
def test_django_update_fields_edge_cases(self):
|
||||||
"""
|
"""
|
||||||
|
|
@ -179,6 +189,11 @@ class SimpleModelTest(TestCase):
|
||||||
{"boolean": ["False", "True"], "integer": ["None", "1"]},
|
{"boolean": ["False", "True"], "integer": ["None", "1"]},
|
||||||
msg="The 2 fields changed are correctly logged",
|
msg="The 2 fields changed are correctly logged",
|
||||||
)
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
obj.history.get(action=LogEntry.Action.UPDATE).changes_str,
|
||||||
|
"boolean: False → True; integer: None → 1",
|
||||||
|
msg="Changes string is correct",
|
||||||
|
)
|
||||||
|
|
||||||
def test_delete(self):
|
def test_delete(self):
|
||||||
"""Deletion is logged correctly."""
|
"""Deletion is logged correctly."""
|
||||||
|
|
@ -494,6 +509,13 @@ class ManyRelatedModelTest(TestCase):
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_changes_str(self):
|
||||||
|
self.obj.related.add(self.related)
|
||||||
|
log_entry = self.obj.history.first()
|
||||||
|
self.assertEqual(
|
||||||
|
log_entry.changes_str, f"related: add {[smart_str(self.related)]}"
|
||||||
|
)
|
||||||
|
|
||||||
def test_adding_existing_related_obj(self):
|
def test_adding_existing_related_obj(self):
|
||||||
self.obj.related.add(self.related)
|
self.obj.related.add(self.related)
|
||||||
log_entry = self.obj.history.first()
|
log_entry = self.obj.history.first()
|
||||||
|
|
@ -725,6 +747,11 @@ class SimpleIncludeModelTest(TestCase):
|
||||||
{"label": ["Initial label", "New label"]},
|
{"label": ["Initial label", "New label"]},
|
||||||
msg="Only the label was logged, regardless of multiple entries in `update_fields`",
|
msg="Only the label was logged, regardless of multiple entries in `update_fields`",
|
||||||
)
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
obj.history.get(action=LogEntry.Action.UPDATE).changes_str,
|
||||||
|
"label: Initial label → New label",
|
||||||
|
msg="Changes string is correct",
|
||||||
|
)
|
||||||
|
|
||||||
def test_register_include_fields(self):
|
def test_register_include_fields(self):
|
||||||
sim = SimpleIncludeModel(label="Include model", text="Looong text")
|
sim = SimpleIncludeModel(label="Include model", text="Looong text")
|
||||||
|
|
@ -2061,6 +2088,11 @@ class JSONModelTest(TestCase):
|
||||||
{"json": ["{}", '{"quantity": "1"}']},
|
{"json": ["{}", '{"quantity": "1"}']},
|
||||||
msg="The change is correctly logged",
|
msg="The change is correctly logged",
|
||||||
)
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
history.changes_str,
|
||||||
|
'json: {} → {"quantity": "1"}',
|
||||||
|
msg="Changes string is correct",
|
||||||
|
)
|
||||||
|
|
||||||
def test_update_with_no_changes(self):
|
def test_update_with_no_changes(self):
|
||||||
"""No changes are logged."""
|
"""No changes are logged."""
|
||||||
|
|
@ -2697,6 +2729,7 @@ class TestAccessLog(TestCase):
|
||||||
)
|
)
|
||||||
self.assertIsNone(log_entry.changes)
|
self.assertIsNone(log_entry.changes)
|
||||||
self.assertEqual(log_entry.changes_dict, {})
|
self.assertEqual(log_entry.changes_dict, {})
|
||||||
|
self.assertEqual(log_entry.changes_str, "")
|
||||||
|
|
||||||
|
|
||||||
class SignalTests(TestCase):
|
class SignalTests(TestCase):
|
||||||
|
|
@ -3120,6 +3153,9 @@ class BaseManagerSettingTest(TestCase):
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
log_entry.changes_str, f"m2m_related: add {[smart_str(obj_two)]}"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class TestMaskStr(TestCase):
|
class TestMaskStr(TestCase):
|
||||||
|
|
|
||||||
|
|
@ -76,7 +76,7 @@ You can also add log-access to function base views, as the following example ill
|
||||||
|
|
||||||
Fields that are excluded will not trigger saving a new log entry and will not show up in the recorded changes.
|
Fields that are excluded will not trigger saving a new log entry and will not show up in the recorded changes.
|
||||||
|
|
||||||
To exclude specific fields from the log you can pass ``include_fields`` resp. ``exclude_fields`` to the ``register``
|
To exclude specific fields from the log you can pass ``include_fields`` or ``exclude_fields`` to the ``register``
|
||||||
method. If ``exclude_fields`` is specified the fields with the given names will not be included in the generated log
|
method. If ``exclude_fields`` is specified the fields with the given names will not be included in the generated log
|
||||||
entries. If ``include_fields`` is specified only the fields with the given names will be included in the generated log
|
entries. If ``include_fields`` is specified only the fields with the given names will be included in the generated log
|
||||||
entries. Explicitly excluding fields through ``exclude_fields`` takes precedence over specifying which fields to
|
entries. Explicitly excluding fields through ``exclude_fields`` takes precedence over specifying which fields to
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue