2020-08-07 21:39:24 +02:00
|
|
|
# Generated by Django 2.2.14 on 2020-08-07 19:13
|
|
|
|
|
|
|
|
import json
|
|
|
|
|
|
|
|
from django.db import migrations
|
|
|
|
from django.db.backends.postgresql.schema import DatabaseSchemaEditor
|
|
|
|
from django.db.migrations.state import StateApps
|
|
|
|
|
|
|
|
|
|
|
|
def update_realmauditlog_values(apps: StateApps, schema_editor: DatabaseSchemaEditor) -> None:
|
|
|
|
"""
|
|
|
|
This migration fixes two issues with the RealmAuditLog format for certain event types:
|
|
|
|
* The notifications_stream and signup_notifications_stream fields had the
|
|
|
|
Stream objects passed into `ujson.dumps()` and thus marshalled as a giant
|
|
|
|
JSON object, when the intent was to store the stream ID.
|
|
|
|
* The default_sending_stream would also been marshalled wrong, but are part
|
|
|
|
of a feature that nobody should be using, so we simply assert that's the case.
|
|
|
|
* Changes the structure of the extra_data JSON dictionaries for those
|
|
|
|
RealmAuditLog entries with a sub-property field from:
|
|
|
|
{
|
|
|
|
OLD_VALUE: {"property": property, "value": old_value},
|
|
|
|
NEW_VALUE: {"property": property, "value": new_value},
|
|
|
|
}
|
|
|
|
|
|
|
|
to the more natural:
|
|
|
|
|
|
|
|
{
|
|
|
|
OLD_VALUE: old_value,
|
|
|
|
NEW_VALUE: new_value,
|
|
|
|
"property": property,
|
|
|
|
}
|
|
|
|
"""
|
2021-02-12 08:20:45 +01:00
|
|
|
RealmAuditLog = apps.get_model("zerver", "RealmAuditLog")
|
2020-08-07 21:39:24 +02:00
|
|
|
# Constants from models.py
|
|
|
|
USER_DEFAULT_SENDING_STREAM_CHANGED = 129
|
|
|
|
USER_DEFAULT_REGISTER_STREAM_CHANGED = 130
|
|
|
|
USER_DEFAULT_ALL_PUBLIC_STREAMS_CHANGED = 131
|
2021-09-07 19:33:26 +02:00
|
|
|
# Note that this was renamed to USER_SETTING_CHANGED sometime
|
|
|
|
# after this migration; we preserve the original name here to
|
|
|
|
# highlight that as of this migration, only notification settings
|
|
|
|
# had RealmAuditLog entries for changes.
|
2020-08-07 21:39:24 +02:00
|
|
|
USER_NOTIFICATION_SETTINGS_CHANGED = 132
|
|
|
|
REALM_PROPERTY_CHANGED = 207
|
|
|
|
SUBSCRIPTION_PROPERTY_CHANGED = 304
|
2021-02-12 08:20:45 +01:00
|
|
|
OLD_VALUE = "1"
|
|
|
|
NEW_VALUE = "2"
|
2020-08-07 21:39:24 +02:00
|
|
|
|
|
|
|
unlikely_event_types = [
|
|
|
|
USER_DEFAULT_SENDING_STREAM_CHANGED,
|
|
|
|
USER_DEFAULT_REGISTER_STREAM_CHANGED,
|
|
|
|
USER_DEFAULT_ALL_PUBLIC_STREAMS_CHANGED,
|
|
|
|
]
|
|
|
|
# These 3 event types are the ones that used a format with
|
|
|
|
# OLD_VALUE containing a dictionary with a `property` key.
|
|
|
|
affected_event_types = [
|
|
|
|
REALM_PROPERTY_CHANGED,
|
|
|
|
USER_NOTIFICATION_SETTINGS_CHANGED,
|
|
|
|
SUBSCRIPTION_PROPERTY_CHANGED,
|
|
|
|
]
|
|
|
|
improperly_marshalled_properties = [
|
2021-02-12 08:20:45 +01:00
|
|
|
"notifications_stream",
|
|
|
|
"signup_notifications_stream",
|
2020-08-07 21:39:24 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
# These are also corrupted but are part of a feature nobody uses,
|
|
|
|
# so it's not worth writing code to fix them.
|
|
|
|
assert not RealmAuditLog.objects.filter(event_type__in=unlikely_event_types).exists()
|
|
|
|
|
|
|
|
for ra in RealmAuditLog.objects.filter(event_type__in=affected_event_types):
|
|
|
|
extra_data = json.loads(ra.extra_data)
|
|
|
|
old_key = extra_data[OLD_VALUE]
|
|
|
|
new_key = extra_data[NEW_VALUE]
|
|
|
|
|
|
|
|
# Skip any already-migrated values in case we're running this
|
|
|
|
# migration a second time.
|
|
|
|
if not isinstance(old_key, dict) and not isinstance(new_key, dict):
|
|
|
|
continue
|
2021-02-12 08:20:45 +01:00
|
|
|
if "value" not in old_key or "value" not in new_key:
|
2020-08-07 21:39:24 +02:00
|
|
|
continue
|
|
|
|
|
|
|
|
old_value = old_key["value"]
|
|
|
|
new_value = new_key["value"]
|
|
|
|
prop = old_key["property"]
|
|
|
|
|
|
|
|
# The `authentication_methods` key is the only event whose
|
|
|
|
# action value type is expected to be a dictionary. That
|
|
|
|
# property is marshalled properly but still wants the second
|
|
|
|
# migration below.
|
2021-02-12 08:20:45 +01:00
|
|
|
if prop != "authentication_methods":
|
2020-08-07 21:39:24 +02:00
|
|
|
# For the other properties, we have `stream` rather than `stream['id']`
|
|
|
|
# in the original extra_data object; the fix is simply to extract
|
|
|
|
# the intended ID field via `value = value['id']`.
|
|
|
|
if isinstance(old_value, dict):
|
|
|
|
assert prop in improperly_marshalled_properties
|
2021-02-12 08:20:45 +01:00
|
|
|
old_value = old_value["id"]
|
2020-08-07 21:39:24 +02:00
|
|
|
if isinstance(new_value, dict):
|
|
|
|
assert prop in improperly_marshalled_properties
|
2021-02-12 08:20:45 +01:00
|
|
|
new_value = new_value["id"]
|
2020-08-07 21:39:24 +02:00
|
|
|
|
|
|
|
# Sanity check that the original event has exactly the keys we expect.
|
2020-09-02 06:20:26 +02:00
|
|
|
assert set(extra_data.keys()) <= {OLD_VALUE, NEW_VALUE}
|
2020-08-07 21:39:24 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
ra.extra_data = json.dumps(
|
|
|
|
{
|
|
|
|
OLD_VALUE: old_value,
|
|
|
|
NEW_VALUE: new_value,
|
|
|
|
"property": prop,
|
|
|
|
}
|
|
|
|
)
|
2020-08-07 21:39:24 +02:00
|
|
|
ra.save(update_fields=["extra_data"])
|
|
|
|
|
|
|
|
|
|
|
|
class Migration(migrations.Migration):
|
|
|
|
|
|
|
|
dependencies = [
|
2021-02-12 08:20:45 +01:00
|
|
|
("zerver", "0297_draft"),
|
2020-08-07 21:39:24 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
operations = [
|
2021-02-12 08:19:30 +01:00
|
|
|
migrations.RunPython(
|
|
|
|
update_realmauditlog_values, reverse_code=migrations.RunPython.noop, elidable=True
|
|
|
|
),
|
2020-08-07 21:39:24 +02:00
|
|
|
]
|