zulip/zerver/migrations/0480_realm_backfill_uuid_an...

70 lines
2.0 KiB
Python

import secrets
import uuid
from django.db import migrations, models
from django.db.backends.base.schema import BaseDatabaseSchemaEditor
from django.db.migrations.state import StateApps
def generate_api_key() -> str:
"""
This is a copy of zerver.lib.utils.generate_api_key. Importing code that's prone
to change in a migration is something we generally avoid, to ensure predictable,
consistent behavior of the migration across time.
"""
api_key = ""
while len(api_key) < 32:
api_key += secrets.token_urlsafe(3 * 9).replace("_", "").replace("-", "")
return api_key[:32]
def generate_realm_uuid_owner_secret() -> str:
token = generate_api_key()
return f"zuliprealm_{token}"
def backfill_realm_uuid_and_secret(
apps: StateApps, schema_editor: BaseDatabaseSchemaEditor
) -> None:
Realm = apps.get_model("zerver", "Realm")
max_id = Realm.objects.aggregate(models.Max("id"))["id__max"]
if max_id is None:
# Nothing to do if there are no realms yet.
return
BATCH_SIZE = 100
lower_bound = 0
while lower_bound < max_id:
realms_to_update = []
for realm in Realm.objects.filter(
id__gt=lower_bound,
id__lte=lower_bound + BATCH_SIZE,
# We're setting uuid and uuid_owner_secret together, so it's enough
# to query for one of them being None.
uuid=None,
).only("id", "uuid", "uuid_owner_secret"):
realm.uuid = uuid.uuid4()
realm.uuid_owner_secret = generate_realm_uuid_owner_secret()
realms_to_update.append(realm)
lower_bound += BATCH_SIZE
Realm.objects.bulk_update(realms_to_update, ["uuid", "uuid_owner_secret"])
class Migration(migrations.Migration):
atomic = False
dependencies = [
("zerver", "0479_realm_uuid_realm_uuid_owner_secret"),
]
operations = [
migrations.RunPython(
backfill_realm_uuid_and_secret, reverse_code=migrations.RunPython.noop
),
]