feat: Implement initial schema and add various API, service, and management command enhancements across the application.

This commit is contained in:
pacnpal
2026-01-01 15:13:01 -05:00
parent c95f99ca10
commit b243b17af7
413 changed files with 11164 additions and 17433 deletions

View File

@@ -102,16 +102,12 @@ class Migration(migrations.Migration):
),
(
"size_acres",
models.DecimalField(
blank=True, decimal_places=2, max_digits=10, null=True
),
models.DecimalField(blank=True, decimal_places=2, max_digits=10, null=True),
),
("website", models.URLField(blank=True)),
(
"average_rating",
models.DecimalField(
blank=True, decimal_places=2, max_digits=3, null=True
),
models.DecimalField(blank=True, decimal_places=2, max_digits=3, null=True),
),
("ride_count", models.IntegerField(blank=True, null=True)),
("coaster_count", models.IntegerField(blank=True, null=True)),
@@ -266,16 +262,12 @@ class Migration(migrations.Migration):
),
(
"size_acres",
models.DecimalField(
blank=True, decimal_places=2, max_digits=10, null=True
),
models.DecimalField(blank=True, decimal_places=2, max_digits=10, null=True),
),
("website", models.URLField(blank=True)),
(
"average_rating",
models.DecimalField(
blank=True, decimal_places=2, max_digits=3, null=True
),
models.DecimalField(blank=True, decimal_places=2, max_digits=3, null=True),
),
("ride_count", models.IntegerField(blank=True, null=True)),
("coaster_count", models.IntegerField(blank=True, null=True)),
@@ -678,9 +670,7 @@ class Migration(migrations.Migration):
),
migrations.AddIndex(
model_name="parklocation",
index=models.Index(
fields=["city", "state"], name="parks_parkl_city_7cc873_idx"
),
index=models.Index(fields=["city", "state"], name="parks_parkl_city_7cc873_idx"),
),
migrations.AlterUniqueTogether(
name="parkreview",

View File

@@ -35,9 +35,7 @@ class Migration(migrations.Migration):
),
(
"state_province",
models.CharField(
blank=True, help_text="State/Province/Region", max_length=100
),
models.CharField(blank=True, help_text="State/Province/Region", max_length=100),
),
(
"country",
@@ -49,9 +47,7 @@ class Migration(migrations.Migration):
),
(
"postal_code",
models.CharField(
blank=True, help_text="ZIP or postal code", max_length=20
),
models.CharField(blank=True, help_text="ZIP or postal code", max_length=20),
),
(
"mailing_address",

View File

@@ -133,21 +133,15 @@ class Migration(migrations.Migration):
),
migrations.AddIndex(
model_name="parkphoto",
index=models.Index(
fields=["park", "is_primary"], name="parks_parkp_park_id_eda26e_idx"
),
index=models.Index(fields=["park", "is_primary"], name="parks_parkp_park_id_eda26e_idx"),
),
migrations.AddIndex(
model_name="parkphoto",
index=models.Index(
fields=["park", "is_approved"], name="parks_parkp_park_id_5fe576_idx"
),
index=models.Index(fields=["park", "is_approved"], name="parks_parkp_park_id_5fe576_idx"),
),
migrations.AddIndex(
model_name="parkphoto",
index=models.Index(
fields=["created_at"], name="parks_parkp_created_033dc3_idx"
),
index=models.Index(fields=["created_at"], name="parks_parkp_created_033dc3_idx"),
),
migrations.AddConstraint(
model_name="parkphoto",

View File

@@ -11,24 +11,29 @@ def populate_computed_fields(apps, schema_editor):
try:
# Use raw SQL to update opening_year from opening_date
schema_editor.execute("""
schema_editor.execute(
"""
UPDATE parks_park
SET opening_year = EXTRACT(YEAR FROM opening_date)
WHERE opening_date IS NOT NULL;
""")
"""
)
# Use raw SQL to populate search_text
# This is a simplified version - we'll populate it with just name and description
schema_editor.execute("""
schema_editor.execute(
"""
UPDATE parks_park
SET search_text = LOWER(
COALESCE(name, '') || ' ' ||
COALESCE(description, '')
);
""")
"""
)
# Update search_text to include operator names using a join
schema_editor.execute("""
schema_editor.execute(
"""
UPDATE parks_park
SET search_text = LOWER(
COALESCE(parks_park.name, '') || ' ' ||
@@ -37,7 +42,8 @@ def populate_computed_fields(apps, schema_editor):
)
FROM parks_company
WHERE parks_park.operator_id = parks_company.id;
""")
"""
)
finally:
# Re-enable pghistory triggers
@@ -46,8 +52,8 @@ def populate_computed_fields(apps, schema_editor):
def reverse_populate_computed_fields(apps, schema_editor):
"""Clear computed fields (reverse operation)"""
Park = apps.get_model('parks', 'Park')
Park.objects.update(opening_year=None, search_text='')
Park = apps.get_model("parks", "Park")
Park.objects.update(opening_year=None, search_text="")
class Migration(migrations.Migration):

View File

@@ -13,37 +13,34 @@ class Migration(migrations.Migration):
# Composite indexes for common filter combinations
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_status_park_type_idx ON parks_park (status, park_type);",
reverse_sql="DROP INDEX IF EXISTS parks_park_status_park_type_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_status_park_type_idx;",
),
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_opening_year_status_idx ON parks_park (opening_year, status) WHERE opening_year IS NOT NULL;",
reverse_sql="DROP INDEX IF EXISTS parks_park_opening_year_status_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_opening_year_status_idx;",
),
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_size_rating_idx ON parks_park (size_acres, average_rating) WHERE size_acres IS NOT NULL AND average_rating IS NOT NULL;",
reverse_sql="DROP INDEX IF EXISTS parks_park_size_rating_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_size_rating_idx;",
),
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_ride_coaster_count_idx ON parks_park (ride_count, coaster_count) WHERE ride_count IS NOT NULL AND coaster_count IS NOT NULL;",
reverse_sql="DROP INDEX IF EXISTS parks_park_ride_coaster_count_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_ride_coaster_count_idx;",
),
# Full-text search index for search_text field
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_search_text_gin_idx ON parks_park USING gin(to_tsvector('english', search_text));",
reverse_sql="DROP INDEX IF EXISTS parks_park_search_text_gin_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_search_text_gin_idx;",
),
# Trigram index for fuzzy search on search_text
migrations.RunSQL(
"CREATE EXTENSION IF NOT EXISTS pg_trgm;",
reverse_sql="-- Cannot drop extension as it might be used elsewhere"
reverse_sql="-- Cannot drop extension as it might be used elsewhere",
),
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_search_text_trgm_idx ON parks_park USING gin(search_text gin_trgm_ops);",
reverse_sql="DROP INDEX IF EXISTS parks_park_search_text_trgm_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_search_text_trgm_idx;",
),
# Indexes for location-based filtering (assuming location relationship exists)
migrations.RunSQL(
"""
@@ -51,27 +48,23 @@ class Migration(migrations.Migration):
ON parks_parklocation (country, state)
WHERE country IS NOT NULL AND state IS NOT NULL;
""",
reverse_sql="DROP INDEX IF EXISTS parks_parklocation_country_state_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_parklocation_country_state_idx;",
),
# Index for operator-based filtering
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_operator_status_idx ON parks_park (operator_id, status);",
reverse_sql="DROP INDEX IF EXISTS parks_park_operator_status_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_operator_status_idx;",
),
# Partial indexes for common status filters
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_operating_parks_idx ON parks_park (name, opening_year) WHERE status IN ('OPERATING', 'CLOSED_TEMP');",
reverse_sql="DROP INDEX IF EXISTS parks_park_operating_parks_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_operating_parks_idx;",
),
# Index for ordering by name (already exists but ensuring it's optimized)
migrations.RunSQL(
"CREATE INDEX IF NOT EXISTS parks_park_name_lower_idx ON parks_park (LOWER(name));",
reverse_sql="DROP INDEX IF EXISTS parks_park_name_lower_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_name_lower_idx;",
),
# Covering index for common query patterns
migrations.RunSQL(
"""
@@ -80,6 +73,6 @@ class Migration(migrations.Migration):
INCLUDE (name, slug, size_acres, average_rating, ride_count, coaster_count, operator_id)
WHERE status IN ('OPERATING', 'CLOSED_TEMP');
""",
reverse_sql="DROP INDEX IF EXISTS parks_park_hybrid_covering_idx;"
reverse_sql="DROP INDEX IF EXISTS parks_park_hybrid_covering_idx;",
),
]

View File

@@ -47,6 +47,6 @@ class Migration(migrations.Migration):
reverse_sql="""
-- This is irreversible, but we can drop and recreate without timezone
DROP FUNCTION IF EXISTS pgtrigger_insert_insert_66883() CASCADE;
"""
""",
),
]

View File

@@ -47,6 +47,6 @@ class Migration(migrations.Migration):
reverse_sql="""
-- This is irreversible, but we can drop and recreate without timezone
DROP FUNCTION IF EXISTS pgtrigger_update_update_19f56() CASCADE;
"""
""",
),
]

View File

@@ -14,7 +14,7 @@ from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('parks', '0022_alter_company_roles_alter_companyevent_roles'),
("parks", "0022_alter_company_roles_alter_companyevent_roles"),
]
operations = [

View File

@@ -11,16 +11,16 @@ from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('parks', '0023_add_company_roles_gin_index'),
("parks", "0023_add_company_roles_gin_index"),
]
operations = [
migrations.AlterField(
model_name='park',
name='timezone',
model_name="park",
name="timezone",
field=models.CharField(
blank=True,
default='UTC',
default="UTC",
help_text="Timezone identifier for park operations (e.g., 'America/New_York')",
max_length=50,
),

View File

@@ -61,16 +61,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="company",
name="description",
field=models.TextField(
blank=True, help_text="Detailed company description"
),
field=models.TextField(blank=True, help_text="Detailed company description"),
),
migrations.AlterField(
model_name="company",
name="founded_year",
field=models.PositiveIntegerField(
blank=True, help_text="Year the company was founded", null=True
),
field=models.PositiveIntegerField(blank=True, help_text="Year the company was founded", null=True),
),
migrations.AlterField(
model_name="company",
@@ -80,16 +76,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="company",
name="parks_count",
field=models.IntegerField(
default=0, help_text="Number of parks operated (auto-calculated)"
),
field=models.IntegerField(default=0, help_text="Number of parks operated (auto-calculated)"),
),
migrations.AlterField(
model_name="company",
name="rides_count",
field=models.IntegerField(
default=0, help_text="Number of rides manufactured (auto-calculated)"
),
field=models.IntegerField(default=0, help_text="Number of rides manufactured (auto-calculated)"),
),
migrations.AlterField(
model_name="company",
@@ -114,9 +106,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="company",
name="slug",
field=models.SlugField(
help_text="URL-friendly identifier", max_length=255, unique=True
),
field=models.SlugField(help_text="URL-friendly identifier", max_length=255, unique=True),
),
migrations.AlterField(
model_name="company",
@@ -126,16 +116,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="companyevent",
name="description",
field=models.TextField(
blank=True, help_text="Detailed company description"
),
field=models.TextField(blank=True, help_text="Detailed company description"),
),
migrations.AlterField(
model_name="companyevent",
name="founded_year",
field=models.PositiveIntegerField(
blank=True, help_text="Year the company was founded", null=True
),
field=models.PositiveIntegerField(blank=True, help_text="Year the company was founded", null=True),
),
migrations.AlterField(
model_name="companyevent",
@@ -145,16 +131,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="companyevent",
name="parks_count",
field=models.IntegerField(
default=0, help_text="Number of parks operated (auto-calculated)"
),
field=models.IntegerField(default=0, help_text="Number of parks operated (auto-calculated)"),
),
migrations.AlterField(
model_name="companyevent",
name="rides_count",
field=models.IntegerField(
default=0, help_text="Number of rides manufactured (auto-calculated)"
),
field=models.IntegerField(default=0, help_text="Number of rides manufactured (auto-calculated)"),
),
migrations.AlterField(
model_name="companyevent",
@@ -179,9 +161,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="companyevent",
name="slug",
field=models.SlugField(
db_index=False, help_text="URL-friendly identifier", max_length=255
),
field=models.SlugField(db_index=False, help_text="URL-friendly identifier", max_length=255),
),
migrations.AlterField(
model_name="companyevent",
@@ -229,9 +209,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="park",
name="coaster_count",
field=models.IntegerField(
blank=True, help_text="Total coaster count", null=True
),
field=models.IntegerField(blank=True, help_text="Total coaster count", null=True),
),
migrations.AlterField(
model_name="park",
@@ -251,16 +229,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="park",
name="operating_season",
field=models.CharField(
blank=True, help_text="Operating season", max_length=255
),
field=models.CharField(blank=True, help_text="Operating season", max_length=255),
),
migrations.AlterField(
model_name="park",
name="ride_count",
field=models.IntegerField(
blank=True, help_text="Total ride count", null=True
),
field=models.IntegerField(blank=True, help_text="Total ride count", null=True),
),
migrations.AlterField(
model_name="park",
@@ -276,9 +250,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="park",
name="slug",
field=models.SlugField(
help_text="URL-friendly identifier", max_length=255, unique=True
),
field=models.SlugField(help_text="URL-friendly identifier", max_length=255, unique=True),
),
migrations.AlterField(
model_name="park",
@@ -300,16 +272,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkarea",
name="closing_date",
field=models.DateField(
blank=True, help_text="Date this area closed (if applicable)", null=True
),
field=models.DateField(blank=True, help_text="Date this area closed (if applicable)", null=True),
),
migrations.AlterField(
model_name="parkarea",
name="description",
field=models.TextField(
blank=True, help_text="Detailed description of the area"
),
field=models.TextField(blank=True, help_text="Detailed description of the area"),
),
migrations.AlterField(
model_name="parkarea",
@@ -319,9 +287,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkarea",
name="opening_date",
field=models.DateField(
blank=True, help_text="Date this area opened", null=True
),
field=models.DateField(blank=True, help_text="Date this area opened", null=True),
),
migrations.AlterField(
model_name="parkarea",
@@ -336,23 +302,17 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkarea",
name="slug",
field=models.SlugField(
help_text="URL-friendly identifier (unique within park)", max_length=255
),
field=models.SlugField(help_text="URL-friendly identifier (unique within park)", max_length=255),
),
migrations.AlterField(
model_name="parkareaevent",
name="closing_date",
field=models.DateField(
blank=True, help_text="Date this area closed (if applicable)", null=True
),
field=models.DateField(blank=True, help_text="Date this area closed (if applicable)", null=True),
),
migrations.AlterField(
model_name="parkareaevent",
name="description",
field=models.TextField(
blank=True, help_text="Detailed description of the area"
),
field=models.TextField(blank=True, help_text="Detailed description of the area"),
),
migrations.AlterField(
model_name="parkareaevent",
@@ -362,9 +322,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkareaevent",
name="opening_date",
field=models.DateField(
blank=True, help_text="Date this area opened", null=True
),
field=models.DateField(blank=True, help_text="Date this area opened", null=True),
),
migrations.AlterField(
model_name="parkareaevent",
@@ -406,9 +364,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkevent",
name="coaster_count",
field=models.IntegerField(
blank=True, help_text="Total coaster count", null=True
),
field=models.IntegerField(blank=True, help_text="Total coaster count", null=True),
),
migrations.AlterField(
model_name="parkevent",
@@ -428,16 +384,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkevent",
name="operating_season",
field=models.CharField(
blank=True, help_text="Operating season", max_length=255
),
field=models.CharField(blank=True, help_text="Operating season", max_length=255),
),
migrations.AlterField(
model_name="parkevent",
name="ride_count",
field=models.IntegerField(
blank=True, help_text="Total ride count", null=True
),
field=models.IntegerField(blank=True, help_text="Total ride count", null=True),
),
migrations.AlterField(
model_name="parkevent",
@@ -453,9 +405,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkevent",
name="slug",
field=models.SlugField(
db_index=False, help_text="URL-friendly identifier", max_length=255
),
field=models.SlugField(db_index=False, help_text="URL-friendly identifier", max_length=255),
),
migrations.AlterField(
model_name="parkevent",
@@ -496,9 +446,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkphoto",
name="caption",
field=models.CharField(
blank=True, help_text="Photo caption or description", max_length=255
),
field=models.CharField(blank=True, help_text="Photo caption or description", max_length=255),
),
migrations.AlterField(
model_name="parkphoto",
@@ -549,9 +497,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkphotoevent",
name="caption",
field=models.CharField(
blank=True, help_text="Photo caption or description", max_length=255
),
field=models.CharField(blank=True, help_text="Photo caption or description", max_length=255),
),
migrations.AlterField(
model_name="parkphotoevent",
@@ -602,16 +548,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkreview",
name="is_published",
field=models.BooleanField(
default=True, help_text="Whether this review is publicly visible"
),
field=models.BooleanField(default=True, help_text="Whether this review is publicly visible"),
),
migrations.AlterField(
model_name="parkreview",
name="moderated_at",
field=models.DateTimeField(
blank=True, help_text="When this review was moderated", null=True
),
field=models.DateTimeField(blank=True, help_text="When this review was moderated", null=True),
),
migrations.AlterField(
model_name="parkreview",
@@ -628,9 +570,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkreview",
name="moderation_notes",
field=models.TextField(
blank=True, help_text="Internal notes from moderators"
),
field=models.TextField(blank=True, help_text="Internal notes from moderators"),
),
migrations.AlterField(
model_name="parkreview",
@@ -681,16 +621,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkreviewevent",
name="is_published",
field=models.BooleanField(
default=True, help_text="Whether this review is publicly visible"
),
field=models.BooleanField(default=True, help_text="Whether this review is publicly visible"),
),
migrations.AlterField(
model_name="parkreviewevent",
name="moderated_at",
field=models.DateTimeField(
blank=True, help_text="When this review was moderated", null=True
),
field=models.DateTimeField(blank=True, help_text="When this review was moderated", null=True),
),
migrations.AlterField(
model_name="parkreviewevent",
@@ -709,9 +645,7 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name="parkreviewevent",
name="moderation_notes",
field=models.TextField(
blank=True, help_text="Internal notes from moderators"
),
field=models.TextField(blank=True, help_text="Internal notes from moderators"),
),
migrations.AlterField(
model_name="parkreviewevent",