Skip to content

Commit

Permalink
chore: add missing indices for fk constraints (#2324)
Browse files Browse the repository at this point in the history
  • Loading branch information
cquintana92 authored Nov 15, 2024
1 parent 2150db3 commit 7172325
Show file tree
Hide file tree
Showing 2 changed files with 178 additions and 0 deletions.
76 changes: 76 additions & 0 deletions app/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -158,6 +158,8 @@ class File(Base, ModelMixin):
path = sa.Column(sa.String(128), unique=True, nullable=False)
user_id = sa.Column(sa.ForeignKey("users.id", ondelete="cascade"), nullable=True)

__table_args__ = (sa.Index("ix_file_user_id", "user_id"),)

def get_url(self, expires_in=3600):
return s3.get_url(self.path, expires_in)

Expand Down Expand Up @@ -319,6 +321,8 @@ class HibpNotifiedAlias(Base, ModelMixin):

notified_at = sa.Column(ArrowType, default=arrow.utcnow, nullable=False)

__table_args__ = (sa.Index("ix_hibp_notified_alias_user_id", "user_id"),)


class Fido(Base, ModelMixin):
__tablename__ = "fido"
Expand All @@ -333,6 +337,8 @@ class Fido(Base, ModelMixin):
name = sa.Column(sa.String(128), nullable=False, unique=False)
user_id = sa.Column(sa.ForeignKey("users.id", ondelete="cascade"), nullable=True)

__table_args__ = (sa.Index("ix_fido_user_id", "user_id"),)


class User(Base, ModelMixin, UserMixin, PasswordOracle):
__tablename__ = "users"
Expand Down Expand Up @@ -566,6 +572,10 @@ class User(Base, ModelMixin, UserMixin, PasswordOracle):
),
sa.Index("ix_users_delete_on", delete_on),
sa.Index("ix_users_default_mailbox_id", default_mailbox_id),
sa.Index(
"ix_users_default_alias_custom_domain_id", default_alias_custom_domain_id
),
sa.Index("ix_users_profile_picture_id", profile_picture_id),
)

@property
Expand Down Expand Up @@ -1222,6 +1232,8 @@ class ActivationCode(Base, ModelMixin):

expired = sa.Column(ArrowType, nullable=False, default=_expiration_1h)

__table_args__ = (sa.Index("ix_activation_code_user_id", "user_id"),)

def is_expired(self):
return self.expired < arrow.now()

Expand All @@ -1238,6 +1250,8 @@ class ResetPasswordCode(Base, ModelMixin):

expired = sa.Column(ArrowType, nullable=False, default=_expiration_1h)

__table_args__ = (sa.Index("ix_reset_password_code_user_id", "user_id"),)

def is_expired(self):
return self.expired < arrow.now()

Expand Down Expand Up @@ -1280,6 +1294,8 @@ class MfaBrowser(Base, ModelMixin):

user = orm.relationship(User)

__table_args__ = (sa.Index("ix_mfa_browser_user_id", "user_id"),)

@classmethod
def create_new(cls, user, token_length=64) -> "MfaBrowser":
found = False
Expand Down Expand Up @@ -1338,6 +1354,12 @@ class Client(Base, ModelMixin):
user = orm.relationship(User)
referral = orm.relationship("Referral")

__table_args__ = (
sa.Index("ix_client_user_id", "user_id"),
sa.Index("ix_client_icon_id", "icon_id"),
sa.Index("ix_client_referral_id", "referral_id"),
)

def nb_user(self):
return ClientUser.filter_by(client_id=self.id).count()

Expand Down Expand Up @@ -1386,6 +1408,8 @@ class RedirectUri(Base, ModelMixin):

client = orm.relationship(Client, backref="redirect_uris")

__table_args__ = (sa.Index("ix_redirect_uri_client_id", "client_id"),)


class AuthorizationCode(Base, ModelMixin):
__tablename__ = "authorization_code"
Expand All @@ -1407,6 +1431,11 @@ class AuthorizationCode(Base, ModelMixin):

expired = sa.Column(ArrowType, nullable=False, default=_expiration_5m)

__table_args__ = (
sa.Index("ix_authorization_code_client_id", "client_id"),
sa.Index("ix_authorization_code_user_id", "user_id"),
)

def is_expired(self):
return self.expired < arrow.now()

Expand All @@ -1429,6 +1458,11 @@ class OauthToken(Base, ModelMixin):

expired = sa.Column(ArrowType, nullable=False, default=_expiration_1h)

__table_args__ = (
sa.Index("ix_oauth_token_user_id", "user_id"),
sa.Index("ix_oauth_token_client_id", "client_id"),
)

def is_expired(self):
return self.expired < arrow.now()

Expand Down Expand Up @@ -1582,6 +1616,7 @@ class Alias(Base, ModelMixin):
postgresql_ops={"note": "gin_trgm_ops"},
postgresql_using="gin",
),
Index("ix_alias_original_owner_id", "original_owner_id"),
)

user = orm.relationship(User, foreign_keys=[user_id])
Expand Down Expand Up @@ -2079,6 +2114,7 @@ class EmailLog(Base, ModelMixin):
Index("ix_email_log_created_at", "created_at"),
Index("ix_email_log_mailbox_id", "mailbox_id"),
Index("ix_email_log_bounced_mailbox_id", "bounced_mailbox_id"),
Index("ix_email_log_refused_email_id", "refused_email_id"),
)

user_id = sa.Column(
Expand Down Expand Up @@ -2355,6 +2391,7 @@ class AliasUsedOn(Base, ModelMixin):

__table_args__ = (
sa.UniqueConstraint("alias_id", "hostname", name="uq_alias_used"),
sa.Index("ix_alias_used_on_user_id", "user_id"),
)

alias_id = sa.Column(
Expand All @@ -2381,6 +2418,11 @@ class ApiKey(Base, ModelMixin):

user = orm.relationship(User)

__table_args__ = (
sa.Index("ix_api_key_code", "code"),
sa.Index("ix_api_key_user_id", "user_id"),
)

@classmethod
def create(cls, user_id, name=None, **kwargs):
code = random_string(60)
Expand Down Expand Up @@ -2539,6 +2581,7 @@ class AutoCreateRule(Base, ModelMixin):
sa.UniqueConstraint(
"custom_domain_id", "order", name="uq_auto_create_rule_order"
),
sa.Index("ix_auto_create_rule_custom_domain_id", "custom_domain_id"),
)

custom_domain_id = sa.Column(
Expand Down Expand Up @@ -2582,6 +2625,7 @@ class DomainDeletedAlias(Base, ModelMixin):

__table_args__ = (
sa.UniqueConstraint("domain_id", "email", name="uq_domain_trash"),
sa.Index("ix_domain_deleted_alias_user_id", "user_id"),
)

email = sa.Column(sa.String(256), nullable=False)
Expand Down Expand Up @@ -2642,6 +2686,8 @@ class Coupon(Base, ModelMixin):
# a coupon can have an expiration
expires_date = sa.Column(ArrowType, nullable=True)

__table_args__ = (sa.Index("ix_coupon_used_by_user_id", "used_by_user_id"),)


class Directory(Base, ModelMixin):
__tablename__ = "directory"
Expand All @@ -2656,6 +2702,8 @@ class Directory(Base, ModelMixin):
"Mailbox", secondary="directory_mailbox", lazy="joined"
)

__table_args__ = (sa.Index("ix_directory_user_id", "user_id"),)

@property
def mailboxes(self):
if self._mailboxes:
Expand Down Expand Up @@ -2897,6 +2945,8 @@ class RefusedEmail(Base, ModelMixin):
# toggle this when email content (stored at full_report_path & path are deleted)
deleted = sa.Column(sa.Boolean, nullable=False, default=False, server_default="0")

__table_args__ = (sa.Index("ix_refused_email_user_id", "user_id"),)

def get_url(self, expires_in=3600):
if self.path:
return s3.get_url(self.path, expires_in)
Expand All @@ -2919,6 +2969,8 @@ class Referral(Base, ModelMixin):

user = orm.relationship(User, foreign_keys=[user_id], backref="referrals")

__table_args__ = (sa.Index("ix_referral_user_id", "user_id"),)

@property
def nb_user(self) -> int:
return User.filter_by(referral_id=self.id, activated=True).count()
Expand Down Expand Up @@ -2958,6 +3010,8 @@ class SentAlert(Base, ModelMixin):
to_email = sa.Column(sa.String(256), nullable=False)
alert_type = sa.Column(sa.String(256), nullable=False)

__table_args__ = (sa.Index("ix_sent_alert_user_id", "user_id"),)


class AliasMailbox(Base, ModelMixin):
__tablename__ = "alias_mailbox"
Expand Down Expand Up @@ -3203,6 +3257,11 @@ class BatchImport(Base, ModelMixin):
file = orm.relationship(File)
user = orm.relationship(User)

__table_args__ = (
sa.Index("ix_batch_import_file_id", "file_id"),
sa.Index("ix_batch_import_user_id", "user_id"),
)

def nb_alias(self):
return Alias.filter_by(batch_import_id=self.id).count()

Expand All @@ -3223,6 +3282,7 @@ class AuthorizedAddress(Base, ModelMixin):

__table_args__ = (
sa.UniqueConstraint("mailbox_id", "email", name="uq_authorize_address"),
sa.Index("ix_authorized_address_user_id", "user_id"),
)

mailbox = orm.relationship(Mailbox, backref="authorized_addresses")
Expand Down Expand Up @@ -3364,6 +3424,8 @@ class Payout(Base, ModelMixin):

user = orm.relationship(User)

__table_args__ = (sa.Index("ix_payout_user_id", "user_id"),)


class IgnoredEmail(Base, ModelMixin):
"""If an email has mail_from and rcpt_to present in this table, discard it by returning 250 status."""
Expand Down Expand Up @@ -3465,6 +3527,8 @@ class PhoneReservation(Base, ModelMixin):
start = sa.Column(ArrowType, nullable=False)
end = sa.Column(ArrowType, nullable=False)

__table_args__ = (sa.Index("ix_phone_reservation_user_id", "user_id"),)


class PhoneMessage(Base, ModelMixin):
__tablename__ = "phone_message"
Expand Down Expand Up @@ -3639,6 +3703,11 @@ class ProviderComplaint(Base, ModelMixin):
user = orm.relationship(User, foreign_keys=[user_id])
refused_email = orm.relationship(RefusedEmail, foreign_keys=[refused_email_id])

__table_args__ = (
sa.Index("ix_provider_complaint_user_id", "user_id"),
sa.Index("ix_provider_complaint_refused_email_id", "refused_email_id"),
)


class PartnerApiToken(Base, ModelMixin):
__tablename__ = "partner_api_token"
Expand Down Expand Up @@ -3762,6 +3831,8 @@ class NewsletterUser(Base, ModelMixin):
user = orm.relationship(User)
newsletter = orm.relationship(Newsletter)

__table_args__ = (sa.Index("ix_newsletter_user_user_id", "user_id"),)


class ApiToCookieToken(Base, ModelMixin):
__tablename__ = "api_cookie_token"
Expand All @@ -3772,6 +3843,11 @@ class ApiToCookieToken(Base, ModelMixin):
user = orm.relationship(User)
api_key = orm.relationship(ApiKey)

__table_args__ = (
sa.Index("ix_api_to_cookie_token_api_key_id", "api_key_id"),
sa.Index("ix_api_to_cookie_token_user_id", "user_id"),
)

@classmethod
def create(cls, **kwargs):
code = secrets.token_urlsafe(32)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
"""add missing indices for fk constraints
Revision ID: 0f3ee15b0014
Revises: 12274da2299f
Create Date: 2024-11-15 12:29:10.739938
"""
import sqlalchemy_utils
from alembic import op
import sqlalchemy as sa


# revision identifiers, used by Alembic.
revision = '0f3ee15b0014'
down_revision = '12274da2299f'
branch_labels = None
depends_on = None


def upgrade():
with op.get_context().autocommit_block():
op.create_index('ix_activation_code_user_id', 'activation_code', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_alias_original_owner_id', 'alias', ['original_owner_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_alias_used_on_user_id', 'alias_used_on', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_api_to_cookie_token_api_key_id', 'api_cookie_token', ['api_key_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_api_to_cookie_token_user_id', 'api_cookie_token', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_api_key_code', 'api_key', ['code'], unique=False, postgresql_concurrently=True)
op.create_index('ix_api_key_user_id', 'api_key', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_authorization_code_client_id', 'authorization_code', ['client_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_authorization_code_user_id', 'authorization_code', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_authorized_address_user_id', 'authorized_address', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_auto_create_rule_custom_domain_id', 'auto_create_rule', ['custom_domain_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_batch_import_file_id', 'batch_import', ['file_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_batch_import_user_id', 'batch_import', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_client_icon_id', 'client', ['icon_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_client_referral_id', 'client', ['referral_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_client_user_id', 'client', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_coupon_used_by_user_id', 'coupon', ['used_by_user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_directory_user_id', 'directory', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_domain_deleted_alias_user_id', 'domain_deleted_alias', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_email_log_refused_email_id', 'email_log', ['refused_email_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_fido_user_id', 'fido', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_file_user_id', 'file', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_hibp_notified_alias_user_id', 'hibp_notified_alias', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_mfa_browser_user_id', 'mfa_browser', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_newsletter_user_user_id', 'newsletter_user', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_oauth_token_client_id', 'oauth_token', ['client_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_oauth_token_user_id', 'oauth_token', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_payout_user_id', 'payout', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_phone_reservation_user_id', 'phone_reservation', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_provider_complaint_refused_email_id', 'provider_complaint', ['refused_email_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_provider_complaint_user_id', 'provider_complaint', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_redirect_uri_client_id', 'redirect_uri', ['client_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_referral_user_id', 'referral', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_refused_email_user_id', 'refused_email', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_reset_password_code_user_id', 'reset_password_code', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_sent_alert_user_id', 'sent_alert', ['user_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_users_default_alias_custom_domain_id', 'users', ['default_alias_custom_domain_id'], unique=False, postgresql_concurrently=True)
op.create_index('ix_users_profile_picture_id', 'users', ['profile_picture_id'], unique=False, postgresql_concurrently=True)



def downgrade():
with op.get_context().autocommit_block():
op.drop_index('ix_users_profile_picture_id', table_name='users')
op.drop_index('ix_users_default_alias_custom_domain_id', table_name='users')
op.drop_index('ix_sent_alert_user_id', table_name='sent_alert')
op.drop_index('ix_reset_password_code_user_id', table_name='reset_password_code')
op.drop_index('ix_refused_email_user_id', table_name='refused_email')
op.drop_index('ix_referral_user_id', table_name='referral')
op.drop_index('ix_redirect_uri_client_id', table_name='redirect_uri')
op.drop_index('ix_provider_complaint_user_id', table_name='provider_complaint')
op.drop_index('ix_provider_complaint_refused_email_id', table_name='provider_complaint')
op.drop_index('ix_phone_reservation_user_id', table_name='phone_reservation')
op.drop_index('ix_payout_user_id', table_name='payout')
op.drop_index('ix_oauth_token_user_id', table_name='oauth_token')
op.drop_index('ix_oauth_token_client_id', table_name='oauth_token')
op.drop_index('ix_newsletter_user_user_id', table_name='newsletter_user')
op.drop_index('ix_mfa_browser_user_id', table_name='mfa_browser')
op.drop_index('ix_hibp_notified_alias_user_id', table_name='hibp_notified_alias')
op.drop_index('ix_file_user_id', table_name='file')
op.drop_index('ix_fido_user_id', table_name='fido')
op.drop_index('ix_email_log_refused_email_id', table_name='email_log')
op.drop_index('ix_domain_deleted_alias_user_id', table_name='domain_deleted_alias')
op.drop_index('ix_directory_user_id', table_name='directory')
op.drop_index('ix_coupon_used_by_user_id', table_name='coupon')
op.drop_index('ix_client_user_id', table_name='client')
op.drop_index('ix_client_referral_id', table_name='client')
op.drop_index('ix_client_icon_id', table_name='client')
op.drop_index('ix_batch_import_user_id', table_name='batch_import')
op.drop_index('ix_batch_import_file_id', table_name='batch_import')
op.drop_index('ix_auto_create_rule_custom_domain_id', table_name='auto_create_rule')
op.drop_index('ix_authorized_address_user_id', table_name='authorized_address')
op.drop_index('ix_authorization_code_user_id', table_name='authorization_code')
op.drop_index('ix_authorization_code_client_id', table_name='authorization_code')
op.drop_index('ix_api_key_user_id', table_name='api_key')
op.drop_index('ix_api_key_code', table_name='api_key')
op.drop_index('ix_api_to_cookie_token_user_id', table_name='api_cookie_token')
op.drop_index('ix_api_to_cookie_token_api_key_id', table_name='api_cookie_token')
op.drop_index('ix_alias_used_on_user_id', table_name='alias_used_on')
op.drop_index('ix_alias_original_owner_id', table_name='alias')
op.drop_index('ix_activation_code_user_id', table_name='activation_code')

0 comments on commit 7172325

Please sign in to comment.