feat: platform-aware storefront routing and billing improvements
Overhaul storefront URL routing to be platform-aware:
- Dev: /platforms/{code}/storefront/{store_code}/
- Prod: subdomain.platform.lu/ (internally rewritten to /storefront/)
- Add subdomain detection in PlatformContextMiddleware
- Add /storefront/ path rewrite for prod mode (subdomain/custom domain)
- Remove all silent platform fallbacks (platform_id=1)
- Add require_platform dependency for clean endpoint validation
- Update route registration, templates, module definitions, base_url calc
- Update StoreContextMiddleware for /storefront/ path detection
- Remove /stores/ from FrontendDetector STOREFRONT_PATH_PREFIXES
Billing service improvements:
- Add store_platform_sync_service to keep store_platforms in sync
- Make tier lookups platform-aware across billing services
- Add tiers for all platforms in seed data
- Add demo subscriptions to seed
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -65,10 +65,11 @@ class TestFrontendDetectorStore:
|
||||
result = FrontendDetector.detect(host="omsflow.lu", path="/store/dashboard/analytics")
|
||||
assert result == FrontendType.STORE
|
||||
|
||||
def test_stores_plural_not_store_dashboard(self):
|
||||
"""Test that /stores/ path is NOT store dashboard (it's storefront)."""
|
||||
def test_stores_plural_is_not_storefront(self):
|
||||
"""Test that /stores/ path is NOT storefront (old pattern removed)."""
|
||||
result = FrontendDetector.detect(host="localhost", path="/stores/orion/storefront")
|
||||
assert result == FrontendType.STOREFRONT
|
||||
# /stores/ is no longer a storefront signal; /storefront/ is the canonical prefix
|
||||
assert result != FrontendType.STOREFRONT
|
||||
|
||||
|
||||
@pytest.mark.unit
|
||||
@@ -85,9 +86,9 @@ class TestFrontendDetectorStorefront:
|
||||
result = FrontendDetector.detect(host="localhost", path="/api/v1/storefront/cart")
|
||||
assert result == FrontendType.STOREFRONT
|
||||
|
||||
def test_detect_storefront_from_stores_path(self):
|
||||
"""Test storefront detection from /stores/ path (path-based store access)."""
|
||||
result = FrontendDetector.detect(host="localhost", path="/stores/orion/products")
|
||||
def test_detect_storefront_from_storefront_path(self):
|
||||
"""Test storefront detection from /storefront/ path (path-based store access)."""
|
||||
result = FrontendDetector.detect(host="localhost", path="/storefront/orion/products")
|
||||
assert result == FrontendType.STOREFRONT
|
||||
|
||||
def test_detect_storefront_from_store_subdomain(self):
|
||||
|
||||
@@ -24,7 +24,7 @@ from sqlalchemy.orm import Session
|
||||
|
||||
from app.core.frontend_detector import FrontendDetector
|
||||
from middleware.platform_context import (
|
||||
DEFAULT_PLATFORM_CODE,
|
||||
MAIN_PLATFORM_CODE,
|
||||
PlatformContextManager,
|
||||
PlatformContextMiddleware,
|
||||
get_current_platform,
|
||||
@@ -174,7 +174,7 @@ class TestPlatformContextManager:
|
||||
|
||||
assert context is not None
|
||||
assert context["detection_method"] == "default"
|
||||
assert context["path_prefix"] == DEFAULT_PLATFORM_CODE
|
||||
assert context["path_prefix"] == MAIN_PLATFORM_CODE
|
||||
assert context["clean_path"] == "/about" # No path rewrite for main site
|
||||
|
||||
def test_detect_default_platform_127_0_0_1(self):
|
||||
@@ -187,7 +187,7 @@ class TestPlatformContextManager:
|
||||
|
||||
assert context is not None
|
||||
assert context["detection_method"] == "default"
|
||||
assert context["path_prefix"] == DEFAULT_PLATFORM_CODE
|
||||
assert context["path_prefix"] == MAIN_PLATFORM_CODE
|
||||
|
||||
def test_detect_default_platform_root_path(self):
|
||||
"""Test default platform detection for root path."""
|
||||
@@ -797,8 +797,8 @@ class TestHelperFunctions:
|
||||
assert "Platform not found" in exc_info.value.detail
|
||||
|
||||
def test_default_platform_code_is_main(self):
|
||||
"""Test that DEFAULT_PLATFORM_CODE is 'main'."""
|
||||
assert DEFAULT_PLATFORM_CODE == "main"
|
||||
"""Test that MAIN_PLATFORM_CODE is 'main'."""
|
||||
assert MAIN_PLATFORM_CODE == "main"
|
||||
|
||||
|
||||
@pytest.mark.unit
|
||||
|
||||
@@ -430,51 +430,40 @@ class TestGetSubscription:
|
||||
)
|
||||
assert result is expected_sub
|
||||
|
||||
def test_falls_back_to_store_primary_platform(self):
|
||||
"""Test fallback to get_subscription_for_store when platform sub is None."""
|
||||
def test_no_fallback_when_merchant_subscription_none(self):
|
||||
"""Test no fallback when get_merchant_subscription returns None."""
|
||||
middleware = StorefrontAccessMiddleware(app=None)
|
||||
store = _make_store(store_id=5, merchant_id=10)
|
||||
platform = _make_platform(platform_id=2)
|
||||
request = _make_request(store=store, platform=platform)
|
||||
mock_db = MagicMock()
|
||||
fallback_sub = _make_subscription(tier_code="starter")
|
||||
|
||||
with patch(
|
||||
"app.modules.billing.services.subscription_service.subscription_service"
|
||||
) as mock_svc:
|
||||
mock_svc.get_merchant_subscription.return_value = None
|
||||
mock_svc.get_subscription_for_store.return_value = fallback_sub
|
||||
|
||||
result = middleware._get_subscription(mock_db, store, request)
|
||||
|
||||
mock_svc.get_merchant_subscription.assert_called_once_with(
|
||||
mock_db, 10, 2
|
||||
)
|
||||
mock_svc.get_subscription_for_store.assert_called_once_with(
|
||||
mock_db, 5
|
||||
)
|
||||
assert result is fallback_sub
|
||||
assert result is None
|
||||
|
||||
def test_no_platform_uses_store_fallback(self):
|
||||
"""Test when no platform is detected, falls back to store-based lookup."""
|
||||
def test_no_platform_returns_none(self):
|
||||
"""Test when no platform is detected, returns None (no fallback)."""
|
||||
middleware = StorefrontAccessMiddleware(app=None)
|
||||
store = _make_store(store_id=7)
|
||||
request = _make_request(store=store, platform=None)
|
||||
mock_db = MagicMock()
|
||||
fallback_sub = _make_subscription()
|
||||
|
||||
with patch(
|
||||
"app.modules.billing.services.subscription_service.subscription_service"
|
||||
) as mock_svc:
|
||||
mock_svc.get_subscription_for_store.return_value = fallback_sub
|
||||
|
||||
result = middleware._get_subscription(mock_db, store, request)
|
||||
|
||||
mock_svc.get_merchant_subscription.assert_not_called()
|
||||
mock_svc.get_subscription_for_store.assert_called_once_with(
|
||||
mock_db, 7
|
||||
)
|
||||
assert result is fallback_sub
|
||||
assert result is None
|
||||
|
||||
|
||||
# =============================================================================
|
||||
|
||||
Reference in New Issue
Block a user