diff --git a/backend/celerybeat-schedule b/backend/celerybeat-schedule
index fc8f191f..0c426d4b 100644
Binary files a/backend/celerybeat-schedule and b/backend/celerybeat-schedule differ
diff --git a/backend/create_test_users.py b/backend/create_test_users.py
deleted file mode 100644
index bb45892c..00000000
--- a/backend/create_test_users.py
+++ /dev/null
@@ -1,187 +0,0 @@
-#!/usr/bin/env python
-"""
-Script to create 3 real users with 3 paid packages (Starter, Growth, Scale)
-All accounts will be active and properly configured.
-Email format: plan-name@igny8.com
-"""
-import os
-import django
-import sys
-from decimal import Decimal
-
-# Setup Django
-os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'igny8_core.settings')
-django.setup()
-
-from django.db import transaction
-from igny8_core.auth.models import Plan, Account, User
-from django.utils.text import slugify
-
-# User data - 3 users with 3 different paid plans
-# Email format: plan-name@igny8.com
-USERS_DATA = [
- {
- "email": "starter@igny8.com",
- "username": "starter",
- "first_name": "Starter",
- "last_name": "Account",
- "password": "SecurePass123!@#",
- "plan_slug": "starter", # $89/month
- "account_name": "Starter Account",
- },
- {
- "email": "growth@igny8.com",
- "username": "growth",
- "first_name": "Growth",
- "last_name": "Account",
- "password": "SecurePass123!@#",
- "plan_slug": "growth", # $139/month
- "account_name": "Growth Account",
- },
- {
- "email": "scale@igny8.com",
- "username": "scale",
- "first_name": "Scale",
- "last_name": "Account",
- "password": "SecurePass123!@#",
- "plan_slug": "scale", # $229/month
- "account_name": "Scale Account",
- },
-]
-
-
-def create_user_with_plan(user_data):
- """Create a user with account and assigned plan."""
- try:
- with transaction.atomic():
- # Get the plan
- try:
- plan = Plan.objects.get(slug=user_data['plan_slug'], is_active=True)
- except Plan.DoesNotExist:
- print(f"❌ ERROR: Plan '{user_data['plan_slug']}' not found or inactive!")
- return None
-
- # Check if user already exists
- if User.objects.filter(email=user_data['email']).exists():
- print(f"⚠️ User {user_data['email']} already exists. Updating...")
- existing_user = User.objects.get(email=user_data['email'])
- if existing_user.account:
- existing_user.account.plan = plan
- existing_user.account.status = 'active'
- existing_user.account.save()
- print(f" ✅ Updated account plan to {plan.name} and set status to active")
- return existing_user
-
- # Generate unique account slug
- base_slug = slugify(user_data['account_name'])
- account_slug = base_slug
- counter = 1
- while Account.objects.filter(slug=account_slug).exists():
- account_slug = f"{base_slug}-{counter}"
- counter += 1
-
- # Create user first (without account)
- user = User.objects.create_user(
- username=user_data['username'],
- email=user_data['email'],
- password=user_data['password'],
- first_name=user_data['first_name'],
- last_name=user_data['last_name'],
- account=None, # Will be set after account creation
- role='owner'
- )
-
- # Create account with user as owner and assigned plan
- account = Account.objects.create(
- name=user_data['account_name'],
- slug=account_slug,
- owner=user,
- plan=plan,
- status='active', # Set to active
- credits=plan.included_credits or 0, # Set initial credits from plan
- )
-
- # Update user to reference the new account
- user.account = account
- user.save()
-
- print(f"✅ Created user: {user.email}")
- print(f" - Name: {user.get_full_name()}")
- print(f" - Username: {user.username}")
- print(f" - Account: {account.name} (slug: {account.slug})")
- print(f" - Plan: {plan.name} (${plan.price}/month)")
- print(f" - Status: {account.status}")
- print(f" - Credits: {account.credits}")
- print(f" - Max Sites: {plan.max_sites}")
- print(f" - Max Users: {plan.max_users}")
- print()
-
- return user
-
- except Exception as e:
- print(f"❌ ERROR creating user {user_data['email']}: {e}")
- import traceback
- traceback.print_exc()
- return None
-
-
-def main():
- """Main function to create all users."""
- print("=" * 80)
- print("Creating 3 Users with Paid Plans")
- print("=" * 80)
- print()
-
- # Verify plans exist
- print("Checking available plans...")
- plans = Plan.objects.filter(is_active=True).order_by('price')
- if plans.count() < 3:
- print(f"⚠️ WARNING: Only {plans.count()} active plan(s) found. Need at least 3.")
- print("Available plans:")
- for p in plans:
- print(f" - {p.slug} (${p.price})")
- print()
- print("Please run import_plans.py first to create the plans.")
- return
-
- print("✅ Found plans:")
- for p in plans:
- print(f" - {p.name} ({p.slug}): ${p.price}/month")
- print()
-
- # Create users
- created_users = []
- for user_data in USERS_DATA:
- user = create_user_with_plan(user_data)
- if user:
- created_users.append(user)
-
- # Summary
- print("=" * 80)
- print("SUMMARY")
- print("=" * 80)
- print(f"Total users created/updated: {len(created_users)}")
- print()
- print("User Login Credentials:")
- print("-" * 80)
- for user_data in USERS_DATA:
- print(f"Email: {user_data['email']}")
- print(f"Password: {user_data['password']}")
- print(f"Plan: {user_data['plan_slug'].title()}")
- print()
-
- print("✅ All users created successfully!")
- print()
- print("You can now log in with any of these accounts at:")
- print("https://app.igny8.com/login")
-
-
-if __name__ == '__main__':
- try:
- main()
- except Exception as e:
- print(f"❌ Fatal error: {e}", file=sys.stderr)
- import traceback
- traceback.print_exc()
- sys.exit(1)
-
diff --git a/backend/db.sqlite3 b/backend/db.sqlite3
deleted file mode 100644
index 71ce8694..00000000
Binary files a/backend/db.sqlite3 and /dev/null differ
diff --git a/backend/igny8_core/ai/engine.py b/backend/igny8_core/ai/engine.py
index 0bed9103..4689eeb1 100644
--- a/backend/igny8_core/ai/engine.py
+++ b/backend/igny8_core/ai/engine.py
@@ -34,6 +34,8 @@ class AIEngine:
return f"{count} task{'s' if count != 1 else ''}"
elif function_name == 'generate_images':
return f"{count} task{'s' if count != 1 else ''}"
+ elif function_name == 'generate_site_structure':
+ return "1 site blueprint"
return f"{count} item{'s' if count != 1 else ''}"
def _build_validation_message(self, function_name: str, payload: dict, count: int, input_description: str) -> str:
@@ -80,6 +82,12 @@ class AIEngine:
total_images = 1 + max_images
return f"Mapping Content for {total_images} Image Prompts"
return f"Mapping Content for Image Prompts"
+ elif function_name == 'generate_site_structure':
+ blueprint_name = ''
+ if isinstance(data, dict):
+ blueprint = data.get('blueprint')
+ blueprint_name = f"“{getattr(blueprint, 'name', '')}”" if blueprint and getattr(blueprint, 'name', None) else ''
+ return f"Preparing site blueprint {blueprint_name}".strip()
return f"Preparing {count} item{'s' if count != 1 else ''}"
def _get_ai_call_message(self, function_name: str, count: int) -> str:
@@ -92,6 +100,8 @@ class AIEngine:
return f"Writing article{'s' if count != 1 else ''} with AI"
elif function_name == 'generate_images':
return f"Creating image{'s' if count != 1 else ''} with AI"
+ elif function_name == 'generate_site_structure':
+ return "Designing complete site architecture"
return f"Processing with AI"
def _get_parse_message(self, function_name: str) -> str:
@@ -104,6 +114,8 @@ class AIEngine:
return "Formatting content"
elif function_name == 'generate_images':
return "Processing images"
+ elif function_name == 'generate_site_structure':
+ return "Compiling site map"
return "Processing results"
def _get_parse_message_with_count(self, function_name: str, count: int) -> str:
@@ -122,6 +134,8 @@ class AIEngine:
if in_article_count > 0:
return f"Writing {in_article_count} In‑article Image Prompts"
return "Writing In‑article Image Prompts"
+ elif function_name == 'generate_site_structure':
+ return f"{count} page blueprint{'s' if count != 1 else ''} mapped"
return f"{count} item{'s' if count != 1 else ''} processed"
def _get_save_message(self, function_name: str, count: int) -> str:
@@ -137,6 +151,8 @@ class AIEngine:
elif function_name == 'generate_image_prompts':
# Count is total prompts created
return f"Assigning {count} Prompts to Dedicated Slots"
+ elif function_name == 'generate_site_structure':
+ return f"Publishing {count} page blueprint{'s' if count != 1 else ''}"
return f"Saving {count} item{'s' if count != 1 else ''}"
def execute(self, fn: BaseAIFunction, payload: dict) -> dict:
@@ -494,6 +510,7 @@ class AIEngine:
'generate_content': 'content_generation',
'generate_image_prompts': 'image_prompt_extraction',
'generate_images': 'image_generation',
+ 'generate_site_structure': 'site_structure_generation',
}
return mapping.get(function_name, function_name)
@@ -554,6 +571,7 @@ class AIEngine:
'generate_content': 'content',
'generate_image_prompts': 'image',
'generate_images': 'image',
+ 'generate_site_structure': 'site_blueprint',
}
return mapping.get(function_name, 'unknown')
diff --git a/backend/igny8_core/ai/functions/__init__.py b/backend/igny8_core/ai/functions/__init__.py
index b308eb38..dbf2d49a 100644
--- a/backend/igny8_core/ai/functions/__init__.py
+++ b/backend/igny8_core/ai/functions/__init__.py
@@ -6,6 +6,7 @@ from igny8_core.ai.functions.generate_ideas import GenerateIdeasFunction
from igny8_core.ai.functions.generate_content import GenerateContentFunction
from igny8_core.ai.functions.generate_images import GenerateImagesFunction, generate_images_core
from igny8_core.ai.functions.generate_image_prompts import GenerateImagePromptsFunction
+from igny8_core.ai.functions.generate_site_structure import GenerateSiteStructureFunction
__all__ = [
'AutoClusterFunction',
@@ -14,4 +15,5 @@ __all__ = [
'GenerateImagesFunction',
'generate_images_core',
'GenerateImagePromptsFunction',
+ 'GenerateSiteStructureFunction',
]
diff --git a/backend/igny8_core/ai/functions/generate_site_structure.py b/backend/igny8_core/ai/functions/generate_site_structure.py
new file mode 100644
index 00000000..91898696
--- /dev/null
+++ b/backend/igny8_core/ai/functions/generate_site_structure.py
@@ -0,0 +1,214 @@
+"""
+Generate Site Structure AI Function
+Phase 3 – Site Builder
+"""
+import json
+import logging
+from typing import Any, Dict, List, Tuple
+
+from django.utils.text import slugify
+
+from igny8_core.ai.base import BaseAIFunction
+from igny8_core.ai.prompts import PromptRegistry
+from igny8_core.business.site_building.models import SiteBlueprint, PageBlueprint
+
+logger = logging.getLogger(__name__)
+
+
+class GenerateSiteStructureFunction(BaseAIFunction):
+ """AI function that turns a business brief into a full site blueprint."""
+
+ def get_name(self) -> str:
+ return 'generate_site_structure'
+
+ def get_metadata(self) -> Dict:
+ metadata = super().get_metadata()
+ metadata.update({
+ 'display_name': 'Generate Site Structure',
+ 'description': 'Create site/page architecture from business brief, objectives, and style guides.',
+ 'phases': {
+ 'INIT': 'Validating blueprint data…',
+ 'PREP': 'Preparing site context…',
+ 'AI_CALL': 'Generating site structure with AI…',
+ 'PARSE': 'Parsing generated blueprint…',
+ 'SAVE': 'Saving pages and blocks…',
+ 'DONE': 'Site structure ready!'
+ }
+ })
+ return metadata
+
+ def validate(self, payload: dict, account=None) -> Dict[str, Any]:
+ if not payload.get('ids'):
+ return {'valid': False, 'error': 'Site blueprint ID is required'}
+ return {'valid': True}
+
+ def prepare(self, payload: dict, account=None) -> Dict[str, Any]:
+ blueprint_ids = payload.get('ids', [])
+ queryset = SiteBlueprint.objects.filter(id__in=blueprint_ids)
+ if account:
+ queryset = queryset.filter(account=account)
+ blueprint = queryset.select_related('account', 'site').prefetch_related('pages').first()
+ if not blueprint:
+ raise ValueError("Site blueprint not found")
+
+ config = blueprint.config_json or {}
+ business_brief = payload.get('business_brief') or config.get('business_brief') or ''
+ objectives = payload.get('objectives') or config.get('objectives') or []
+ style = payload.get('style') or config.get('style') or {}
+
+ return {
+ 'blueprint': blueprint,
+ 'business_brief': business_brief,
+ 'objectives': objectives,
+ 'style': style,
+ }
+
+ def build_prompt(self, data: Dict[str, Any], account=None) -> str:
+ blueprint: SiteBlueprint = data['blueprint']
+ objectives = data.get('objectives') or []
+ objectives_text = '\n'.join(f"- {obj}" for obj in objectives) if isinstance(objectives, list) else objectives
+ style = data.get('style') or {}
+ style_text = json.dumps(style, indent=2) if isinstance(style, dict) and style else str(style)
+
+ existing_pages = [
+ {
+ 'title': page.title,
+ 'slug': page.slug,
+ 'type': page.type,
+ 'status': page.status,
+ }
+ for page in blueprint.pages.all()
+ ]
+
+ context = {
+ 'BUSINESS_BRIEF': data.get('business_brief', ''),
+ 'OBJECTIVES': objectives_text or 'Create a full marketing site with clear navigation.',
+ 'STYLE': style_text or 'Modern, responsive, accessible web design.',
+ 'SITE_INFO': json.dumps({
+ 'site_name': blueprint.name,
+ 'site_description': blueprint.description,
+ 'hosting_type': blueprint.hosting_type,
+ 'existing_pages': existing_pages,
+ 'existing_structure': blueprint.structure_json or {},
+ }, indent=2)
+ }
+
+ return PromptRegistry.get_prompt(
+ 'generate_site_structure',
+ account=account or blueprint.account,
+ context=context
+ )
+
+ def parse_response(self, response: str, step_tracker=None) -> Dict[str, Any]:
+ if not response:
+ raise ValueError("AI response is empty")
+
+ response = response.strip()
+ try:
+ return self._ensure_dict(json.loads(response))
+ except json.JSONDecodeError:
+ logger.warning("Response not valid JSON, attempting to extract JSON object")
+ cleaned = self._extract_json_object(response)
+ if cleaned:
+ return self._ensure_dict(json.loads(cleaned))
+ raise ValueError("Unable to parse AI response into JSON")
+
+ def save_output(
+ self,
+ parsed: Dict[str, Any],
+ original_data: Dict[str, Any],
+ account=None,
+ progress_tracker=None,
+ step_tracker=None
+ ) -> Dict[str, Any]:
+ blueprint: SiteBlueprint = original_data['blueprint']
+ structure = self._ensure_dict(parsed)
+ pages = structure.get('pages', [])
+
+ blueprint.structure_json = structure
+ blueprint.status = 'ready'
+ blueprint.save(update_fields=['structure_json', 'status', 'updated_at'])
+
+ created, updated, deleted = self._sync_page_blueprints(blueprint, pages)
+
+ message = f"Pages synced (created: {created}, updated: {updated}, deleted: {deleted})"
+ logger.info("[GenerateSiteStructure] %s for blueprint %s", message, blueprint.id)
+
+ return {
+ 'success': True,
+ 'count': created + updated,
+ 'site_blueprint_id': blueprint.id,
+ 'pages_created': created,
+ 'pages_updated': updated,
+ 'pages_deleted': deleted,
+ }
+
+ # Helpers -----------------------------------------------------------------
+
+ def _ensure_dict(self, data: Any) -> Dict[str, Any]:
+ if isinstance(data, dict):
+ return data
+ raise ValueError("AI response must be a JSON object with site metadata")
+
+ def _extract_json_object(self, text: str) -> str:
+ start = text.find('{')
+ end = text.rfind('}')
+ if start != -1 and end != -1 and end > start:
+ return text[start:end + 1]
+ return ''
+
+ def _sync_page_blueprints(self, blueprint: SiteBlueprint, pages: List[Dict[str, Any]]) -> Tuple[int, int, int]:
+ existing = {page.slug: page for page in blueprint.pages.all()}
+ seen_slugs = set()
+ created = updated = 0
+
+ for order, page_data in enumerate(pages or []):
+ slug = page_data.get('slug') or page_data.get('id') or page_data.get('title') or f"page-{order + 1}"
+ slug = slugify(slug) or f"page-{order + 1}"
+ seen_slugs.add(slug)
+
+ defaults = {
+ 'title': page_data.get('title') or page_data.get('name') or slug.replace('-', ' ').title(),
+ 'type': self._map_page_type(page_data.get('type')),
+ 'blocks_json': page_data.get('blocks') or page_data.get('sections') or [],
+ 'status': page_data.get('status') or 'draft',
+ 'order': order,
+ }
+
+ page_obj, created_flag = PageBlueprint.objects.update_or_create(
+ site_blueprint=blueprint,
+ slug=slug,
+ defaults=defaults
+ )
+ if created_flag:
+ created += 1
+ else:
+ updated += 1
+
+ # Delete pages not present in new structure
+ deleted = 0
+ for slug, page in existing.items():
+ if slug not in seen_slugs:
+ page.delete()
+ deleted += 1
+
+ return created, updated, deleted
+
+ def _map_page_type(self, page_type: Any) -> str:
+ allowed = {choice[0] for choice in PageBlueprint._meta.get_field('type').choices}
+ if isinstance(page_type, str):
+ normalized = page_type.lower()
+ if normalized in allowed:
+ return normalized
+ # Map friendly names
+ mapping = {
+ 'homepage': 'home',
+ 'landing': 'home',
+ 'service': 'services',
+ 'product': 'products',
+ }
+ mapped = mapping.get(normalized)
+ if mapped in allowed:
+ return mapped
+ return 'custom'
+
diff --git a/backend/igny8_core/ai/prompts.py b/backend/igny8_core/ai/prompts.py
index 3f017007..be1de8e9 100644
--- a/backend/igny8_core/ai/prompts.py
+++ b/backend/igny8_core/ai/prompts.py
@@ -238,6 +238,73 @@ OUTPUT FORMAT
Return ONLY the final JSON object.
Do NOT include any comments, formatting, or explanations.""",
+
+ 'site_structure_generation': """You are a senior UX architect and information designer. Use the business brief, objectives, style references, and existing site info to propose a complete multi-page marketing website structure.
+
+INPUT CONTEXT
+==============
+BUSINESS BRIEF:
+[IGNY8_BUSINESS_BRIEF]
+
+PRIMARY OBJECTIVES:
+[IGNY8_OBJECTIVES]
+
+STYLE & BRAND NOTES:
+[IGNY8_STYLE]
+
+SITE INFO / CURRENT STRUCTURE:
+[IGNY8_SITE_INFO]
+
+OUTPUT REQUIREMENTS
+====================
+Return ONE JSON object with the following keys:
+
+{
+ "site": {
+ "name": "...",
+ "primary_navigation": ["home", "services", "about", "contact"],
+ "secondary_navigation": ["blog", "faq"],
+ "hero_message": "High level value statement",
+ "tone": "voice + tone summary"
+ },
+ "pages": [
+ {
+ "slug": "home",
+ "title": "Home",
+ "type": "home | about | services | products | blog | contact | custom",
+ "status": "draft",
+ "objective": "Explain the core brand promise and primary CTA",
+ "primary_cta": "Book a strategy call",
+ "seo": {
+ "meta_title": "...",
+ "meta_description": "..."
+ },
+ "blocks": [
+ {
+ "type": "hero | features | services | stats | testimonials | faq | contact | custom",
+ "heading": "Section headline",
+ "subheading": "Support copy",
+ "layout": "full-width | two-column | cards | carousel",
+ "content": [
+ "Bullet or short paragraph describing what to render in this block"
+ ]
+ }
+ ]
+ }
+ ]
+}
+
+RULES
+=====
+- Include 5–8 pages covering the complete buyer journey (awareness → evaluation → conversion → trust).
+- Every page must have at least 3 blocks with concrete guidance (no placeholders like "Lorem ipsum").
+- Use consistent slug naming, all lowercase with hyphens.
+- Type must match the allowed enum and reflect page intent.
+- Ensure the navigation arrays align with the page list.
+- Focus on practical descriptions that an engineering team can hand off directly to the Site Builder.
+
+Return ONLY valid JSON. No commentary, explanations, or Markdown.
+""",
'image_prompt_extraction': """Extract image prompts from the following article content.
@@ -275,6 +342,7 @@ Make sure each prompt is detailed enough for image generation, describing the vi
'generate_images': 'image_prompt_extraction',
'extract_image_prompts': 'image_prompt_extraction',
'generate_image_prompts': 'image_prompt_extraction',
+ 'generate_site_structure': 'site_structure_generation',
}
@classmethod
diff --git a/backend/igny8_core/ai/registry.py b/backend/igny8_core/ai/registry.py
index fd4da7c2..905db40e 100644
--- a/backend/igny8_core/ai/registry.py
+++ b/backend/igny8_core/ai/registry.py
@@ -94,9 +94,15 @@ def _load_generate_image_prompts():
from igny8_core.ai.functions.generate_image_prompts import GenerateImagePromptsFunction
return GenerateImagePromptsFunction
+def _load_generate_site_structure():
+ """Lazy loader for generate_site_structure function"""
+ from igny8_core.ai.functions.generate_site_structure import GenerateSiteStructureFunction
+ return GenerateSiteStructureFunction
+
register_lazy_function('auto_cluster', _load_auto_cluster)
register_lazy_function('generate_ideas', _load_generate_ideas)
register_lazy_function('generate_content', _load_generate_content)
register_lazy_function('generate_images', _load_generate_images)
register_lazy_function('generate_image_prompts', _load_generate_image_prompts)
+register_lazy_function('generate_site_structure', _load_generate_site_structure)
diff --git a/backend/igny8_core/business/site_building/__init__.py b/backend/igny8_core/business/site_building/__init__.py
index cb479ddf..c337e342 100644
--- a/backend/igny8_core/business/site_building/__init__.py
+++ b/backend/igny8_core/business/site_building/__init__.py
@@ -3,4 +3,4 @@ Site Building Business Logic
Phase 3: Site Builder
"""
-
+default_app_config = 'igny8_core.business.site_building.apps.SiteBuildingConfig'
diff --git a/backend/igny8_core/business/site_building/apps.py b/backend/igny8_core/business/site_building/apps.py
new file mode 100644
index 00000000..51b45348
--- /dev/null
+++ b/backend/igny8_core/business/site_building/apps.py
@@ -0,0 +1,9 @@
+from django.apps import AppConfig
+
+
+class SiteBuildingConfig(AppConfig):
+ default_auto_field = 'django.db.models.BigAutoField'
+ name = 'igny8_core.business.site_building'
+ verbose_name = 'Site Building'
+
+
diff --git a/backend/igny8_core/business/site_building/migrations/0001_initial.py b/backend/igny8_core/business/site_building/migrations/0001_initial.py
new file mode 100644
index 00000000..0f9039f7
--- /dev/null
+++ b/backend/igny8_core/business/site_building/migrations/0001_initial.py
@@ -0,0 +1,94 @@
+from django.db import migrations, models
+import django.db.models.deletion
+import django.core.validators
+
+
+class Migration(migrations.Migration):
+
+ initial = True
+
+ dependencies = [
+ ('igny8_core_auth', '0008_passwordresettoken_alter_industry_options_and_more'),
+ ]
+
+ operations = [
+ migrations.CreateModel(
+ name='SiteBlueprint',
+ fields=[
+ ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+ ('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
+ ('updated_at', models.DateTimeField(auto_now=True)),
+ ('name', models.CharField(help_text='Site name', max_length=255)),
+ ('description', models.TextField(blank=True, help_text='Site description', null=True)),
+ ('config_json', models.JSONField(default=dict, help_text='Wizard configuration: business_type, style, objectives, etc.')),
+ ('structure_json', models.JSONField(default=dict, help_text='AI-generated structure: pages, layout, theme, etc.')),
+ ('status', models.CharField(choices=[('draft', 'Draft'), ('generating', 'Generating'), ('ready', 'Ready'), ('deployed', 'Deployed')], db_index=True, default='draft', help_text='Blueprint status', max_length=20)),
+ ('hosting_type', models.CharField(choices=[('igny8_sites', 'IGNY8 Sites'), ('wordpress', 'WordPress'), ('shopify', 'Shopify'), ('multi', 'Multiple Destinations')], default='igny8_sites', help_text='Target hosting platform', max_length=50)),
+ ('version', models.IntegerField(default=1, help_text='Blueprint version', validators=[django.core.validators.MinValueValidator(1)])),
+ ('deployed_version', models.IntegerField(blank=True, help_text='Currently deployed version', null=True)),
+ ('account', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='siteblueprint_set', to='igny8_core_auth.account')),
+ ('sector', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='siteblueprint_set', to='igny8_core_auth.sector')),
+ ('site', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='siteblueprint_set', to='igny8_core_auth.site')),
+ ],
+ options={
+ 'verbose_name': 'Site Blueprint',
+ 'verbose_name_plural': 'Site Blueprints',
+ 'db_table': 'igny8_site_blueprints',
+ 'ordering': ['-created_at'],
+ },
+ ),
+ migrations.CreateModel(
+ name='PageBlueprint',
+ fields=[
+ ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+ ('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
+ ('updated_at', models.DateTimeField(auto_now=True)),
+ ('slug', models.SlugField(help_text='Page URL slug', max_length=255)),
+ ('title', models.CharField(help_text='Page title', max_length=255)),
+ ('type', models.CharField(choices=[('home', 'Home'), ('about', 'About'), ('services', 'Services'), ('products', 'Products'), ('blog', 'Blog'), ('contact', 'Contact'), ('custom', 'Custom')], default='custom', help_text='Page type', max_length=50)),
+ ('blocks_json', models.JSONField(default=list, help_text="Page content blocks: [{'type': 'hero', 'data': {...}}, ...]")),
+ ('status', models.CharField(choices=[('draft', 'Draft'), ('generating', 'Generating'), ('ready', 'Ready')], db_index=True, default='draft', help_text='Page status', max_length=20)),
+ ('order', models.IntegerField(default=0, help_text='Page order in navigation')),
+ ('account', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='pageblueprint_set', to='igny8_core_auth.account')),
+ ('sector', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='pageblueprint_set', to='igny8_core_auth.sector')),
+ ('site', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='pageblueprint_set', to='igny8_core_auth.site')),
+ ('site_blueprint', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='pages', to='site_building.siteblueprint')),
+ ],
+ options={
+ 'verbose_name': 'Page Blueprint',
+ 'verbose_name_plural': 'Page Blueprints',
+ 'db_table': 'igny8_page_blueprints',
+ 'ordering': ['order', 'created_at'],
+ 'unique_together': {('site_blueprint', 'slug')},
+ },
+ ),
+ migrations.AddIndex(
+ model_name='siteblueprint',
+ index=models.Index(fields=['status'], name='igny8_site__status_247ddc_idx'),
+ ),
+ migrations.AddIndex(
+ model_name='siteblueprint',
+ index=models.Index(fields=['hosting_type'], name='igny8_site__hosting_c4bb41_idx'),
+ ),
+ migrations.AddIndex(
+ model_name='siteblueprint',
+ index=models.Index(fields=['site', 'sector'], name='igny8_site__site_id__5f0a4e_idx'),
+ ),
+ migrations.AddIndex(
+ model_name='siteblueprint',
+ index=models.Index(fields=['account', 'status'], name='igny8_site__account__38f18a_idx'),
+ ),
+ migrations.AddIndex(
+ model_name='pageblueprint',
+ index=models.Index(fields=['site_blueprint', 'status'], name='igny8_page__site_bl_1b5d8b_idx'),
+ ),
+ migrations.AddIndex(
+ model_name='pageblueprint',
+ index=models.Index(fields=['type'], name='igny8_page__type_b11552_idx'),
+ ),
+ migrations.AddIndex(
+ model_name='pageblueprint',
+ index=models.Index(fields=['site_blueprint', 'order'], name='igny8_page__site_bl_7a77d7_idx'),
+ ),
+ ]
+
diff --git a/backend/igny8_core/business/site_building/migrations/__init__.py b/backend/igny8_core/business/site_building/migrations/__init__.py
new file mode 100644
index 00000000..139597f9
--- /dev/null
+++ b/backend/igny8_core/business/site_building/migrations/__init__.py
@@ -0,0 +1,2 @@
+
+
diff --git a/backend/igny8_core/business/site_building/services/__init__.py b/backend/igny8_core/business/site_building/services/__init__.py
index 4ad7ef33..75204e0a 100644
--- a/backend/igny8_core/business/site_building/services/__init__.py
+++ b/backend/igny8_core/business/site_building/services/__init__.py
@@ -2,4 +2,12 @@
Site Building Services
"""
+from igny8_core.business.site_building.services.file_management_service import SiteBuilderFileService
+from igny8_core.business.site_building.services.structure_generation_service import StructureGenerationService
+from igny8_core.business.site_building.services.page_generation_service import PageGenerationService
+__all__ = [
+ 'SiteBuilderFileService',
+ 'StructureGenerationService',
+ 'PageGenerationService',
+]
diff --git a/backend/igny8_core/business/site_building/services/page_generation_service.py b/backend/igny8_core/business/site_building/services/page_generation_service.py
new file mode 100644
index 00000000..79533b9f
--- /dev/null
+++ b/backend/igny8_core/business/site_building/services/page_generation_service.py
@@ -0,0 +1,149 @@
+"""
+Page Generation Service
+Leverages the Writer ContentGenerationService to draft page copy for Site Builder blueprints.
+"""
+import logging
+from typing import Optional
+
+from django.db import transaction
+
+from igny8_core.business.content.models import Tasks
+from igny8_core.business.content.services.content_generation_service import ContentGenerationService
+from igny8_core.business.site_building.models import PageBlueprint
+
+logger = logging.getLogger(__name__)
+
+
+class PageGenerationService:
+ """
+ Thin wrapper that converts Site Builder pages into writer tasks and reuses the
+ existing content generation pipeline. This keeps content authoring logic
+ inside the Writer module while Site Builder focuses on structure.
+ """
+
+ def __init__(self):
+ self.content_service = ContentGenerationService()
+
+ def generate_page_content(self, page_blueprint: PageBlueprint, force_regenerate: bool = False) -> dict:
+ """
+ Generate (or regenerate) content for a single Site Builder page.
+
+ Args:
+ page_blueprint: Target PageBlueprint instance.
+ force_regenerate: If True, resets any temporary task data.
+ """
+ if not page_blueprint:
+ raise ValueError("Page blueprint is required")
+
+ task = self._ensure_task(page_blueprint, force_regenerate=force_regenerate)
+
+ # Mark page as generating before handing off to Writer pipeline
+ page_blueprint.status = 'generating'
+ page_blueprint.save(update_fields=['status', 'updated_at'])
+
+ account = page_blueprint.account
+ logger.info(
+ "[PageGenerationService] Triggering content generation for page %s (task %s)",
+ page_blueprint.id,
+ task.id,
+ )
+ return self.content_service.generate_content([task.id], account)
+
+ def regenerate_page(self, page_blueprint: PageBlueprint) -> dict:
+ """Force regeneration by dropping the cached task metadata."""
+ return self.generate_page_content(page_blueprint, force_regenerate=True)
+
+ # Internal helpers --------------------------------------------------------
+
+ def _ensure_task(self, page_blueprint: PageBlueprint, force_regenerate: bool = False) -> Tasks:
+ """
+ Create or reuse a Writer task that mirrors the given page blueprint.
+ We rely on a deterministic title pattern to keep the mapping lightweight
+ without introducing new relations/migrations.
+ """
+ title = self._build_task_title(page_blueprint)
+ task_qs = Tasks.objects.filter(
+ account=page_blueprint.account,
+ site=page_blueprint.site,
+ sector=page_blueprint.sector,
+ title=title,
+ )
+
+ if force_regenerate:
+ task_qs.delete()
+ else:
+ existing = task_qs.first()
+ if existing:
+ return existing
+
+ return self._create_task_from_page(page_blueprint, title)
+
+ @transaction.atomic
+ def _create_task_from_page(self, page_blueprint: PageBlueprint, title: str) -> Tasks:
+ """Translate blueprint metadata into a Writer task."""
+ description_parts = [
+ f"Site Blueprint: {page_blueprint.site_blueprint.name}",
+ f"Page Type: {page_blueprint.type}",
+ ]
+ hero_block = self._first_block_heading(page_blueprint)
+ if hero_block:
+ description_parts.append(f"Hero/Primary Heading: {hero_block}")
+
+ keywords = self._build_keywords_hint(page_blueprint)
+
+ task = Tasks.objects.create(
+ account=page_blueprint.account,
+ site=page_blueprint.site,
+ sector=page_blueprint.sector,
+ title=title,
+ description="\n".join(filter(None, description_parts)),
+ keywords=keywords,
+ content_structure=self._map_content_structure(page_blueprint.type),
+ content_type='article',
+ status='queued',
+ )
+
+ logger.info(
+ "[PageGenerationService] Created writer task %s for page blueprint %s",
+ task.id,
+ page_blueprint.id,
+ )
+ return task
+
+ def _build_task_title(self, page_blueprint: PageBlueprint) -> str:
+ base = page_blueprint.title or page_blueprint.slug.replace('-', ' ').title()
+ return f"[Site Builder] {base}"
+
+ def _build_keywords_hint(self, page_blueprint: PageBlueprint) -> str:
+ keywords = []
+ if page_blueprint.blocks_json:
+ for block in page_blueprint.blocks_json:
+ heading = block.get('heading') if isinstance(block, dict) else None
+ if heading:
+ keywords.append(heading)
+ keywords.append(page_blueprint.slug.replace('-', ' '))
+ return ", ".join(dict.fromkeys(filter(None, keywords)))
+
+ def _map_content_structure(self, page_type: Optional[str]) -> str:
+ if not page_type:
+ return 'landing_page'
+ mapping = {
+ 'home': 'landing_page',
+ 'about': 'supporting_page',
+ 'services': 'pillar_page',
+ 'products': 'pillar_page',
+ 'blog': 'cluster_hub',
+ 'contact': 'supporting_page',
+ }
+ return mapping.get(page_type.lower(), 'landing_page')
+
+ def _first_block_heading(self, page_blueprint: PageBlueprint) -> Optional[str]:
+ if not page_blueprint.blocks_json:
+ return None
+ for block in page_blueprint.blocks_json:
+ if isinstance(block, dict):
+ heading = block.get('heading') or block.get('title')
+ if heading:
+ return heading
+ return None
+
diff --git a/backend/igny8_core/business/site_building/services/structure_generation_service.py b/backend/igny8_core/business/site_building/services/structure_generation_service.py
new file mode 100644
index 00000000..a486e29b
--- /dev/null
+++ b/backend/igny8_core/business/site_building/services/structure_generation_service.py
@@ -0,0 +1,122 @@
+"""
+Structure Generation Service
+Triggers the AI workflow that maps business briefs to page blueprints.
+"""
+import logging
+from typing import Any, Dict, List, Optional
+
+from django.utils import timezone
+
+from igny8_core.business.billing.exceptions import InsufficientCreditsError
+from igny8_core.business.billing.services.credit_service import CreditService
+from igny8_core.business.site_building.models import SiteBlueprint
+
+logger = logging.getLogger(__name__)
+
+
+class StructureGenerationService:
+ """Orchestrates AI-powered site structure generation."""
+
+ def __init__(self):
+ self.credit_service = CreditService()
+
+ def generate_structure(
+ self,
+ site_blueprint: SiteBlueprint,
+ business_brief: str,
+ objectives: Optional[List[str]] = None,
+ style_preferences: Optional[Dict[str, Any]] = None,
+ metadata: Optional[Dict[str, Any]] = None,
+ ) -> Dict[str, Any]:
+ """
+ Kick off AI structure generation for a single blueprint.
+
+ Args:
+ site_blueprint: Target blueprint instance.
+ business_brief: Business description / positioning statement.
+ objectives: Optional list of goals for the new site.
+ style_preferences: Optional design/style hints.
+ metadata: Additional free-form context.
+ """
+ if not site_blueprint:
+ raise ValueError("Site blueprint is required")
+
+ account = site_blueprint.account
+ objectives = objectives or []
+ style_preferences = style_preferences or {}
+ metadata = metadata or {}
+
+ logger.info(
+ "[StructureGenerationService] Starting generation for blueprint %s (account %s)",
+ site_blueprint.id,
+ getattr(account, 'id', None),
+ )
+
+ # Ensure the account can afford the request
+ try:
+ self.credit_service.check_credits(account, 'site_structure_generation')
+ except InsufficientCreditsError:
+ site_blueprint.status = 'draft'
+ site_blueprint.save(update_fields=['status', 'updated_at'])
+ raise
+
+ # Persist the latest inputs for future regenerations
+ config = site_blueprint.config_json or {}
+ config.update({
+ 'business_brief': business_brief,
+ 'objectives': objectives,
+ 'style': style_preferences,
+ 'last_requested_at': timezone.now().isoformat(),
+ 'metadata': metadata,
+ })
+ site_blueprint.config_json = config
+ site_blueprint.status = 'generating'
+ site_blueprint.save(update_fields=['config_json', 'status', 'updated_at'])
+
+ payload = {
+ 'ids': [site_blueprint.id],
+ 'business_brief': business_brief,
+ 'objectives': objectives,
+ 'style': style_preferences,
+ 'metadata': metadata,
+ }
+
+ return self._dispatch_ai_task(payload, account_id=account.id)
+
+ # Internal helpers --------------------------------------------------------
+
+ def _dispatch_ai_task(self, payload: Dict[str, Any], account_id: int) -> Dict[str, Any]:
+ from igny8_core.ai.tasks import run_ai_task
+
+ try:
+ if hasattr(run_ai_task, 'delay'):
+ async_result = run_ai_task.delay(
+ function_name='generate_site_structure',
+ payload=payload,
+ account_id=account_id
+ )
+ logger.info(
+ "[StructureGenerationService] Queued AI task %s for account %s",
+ async_result.id,
+ account_id,
+ )
+ return {
+ 'success': True,
+ 'task_id': str(async_result.id),
+ 'message': 'Site structure generation queued',
+ }
+
+ # Celery not available – run synchronously
+ logger.warning("[StructureGenerationService] Celery unavailable, running synchronously")
+ return run_ai_task(
+ function_name='generate_site_structure',
+ payload=payload,
+ account_id=account_id
+ )
+ except Exception as exc:
+ logger.error("Failed to dispatch structure generation: %s", exc, exc_info=True)
+ return {
+ 'success': False,
+ 'error': str(exc),
+ }
+
diff --git a/backend/igny8_core/modules/site_builder/__init__.py b/backend/igny8_core/modules/site_builder/__init__.py
new file mode 100644
index 00000000..7ef97d9b
--- /dev/null
+++ b/backend/igny8_core/modules/site_builder/__init__.py
@@ -0,0 +1,5 @@
+"""
+Site Builder module (Phase 3)
+"""
+
+
diff --git a/backend/igny8_core/modules/site_builder/apps.py b/backend/igny8_core/modules/site_builder/apps.py
new file mode 100644
index 00000000..cf80fa8c
--- /dev/null
+++ b/backend/igny8_core/modules/site_builder/apps.py
@@ -0,0 +1,9 @@
+from django.apps import AppConfig
+
+
+class SiteBuilderConfig(AppConfig):
+ default_auto_field = 'django.db.models.BigAutoField'
+ name = 'igny8_core.modules.site_builder'
+ verbose_name = 'Site Builder'
+
+
diff --git a/backend/igny8_core/modules/site_builder/serializers.py b/backend/igny8_core/modules/site_builder/serializers.py
new file mode 100644
index 00000000..0e1715d4
--- /dev/null
+++ b/backend/igny8_core/modules/site_builder/serializers.py
@@ -0,0 +1,78 @@
+from rest_framework import serializers
+
+from igny8_core.business.site_building.models import SiteBlueprint, PageBlueprint
+
+
+class PageBlueprintSerializer(serializers.ModelSerializer):
+ site_blueprint_id = serializers.PrimaryKeyRelatedField(
+ source='site_blueprint',
+ queryset=SiteBlueprint.objects.all(),
+ write_only=True
+ )
+ site_blueprint = serializers.PrimaryKeyRelatedField(read_only=True)
+
+ class Meta:
+ model = PageBlueprint
+ fields = [
+ 'id',
+ 'site_blueprint_id',
+ 'site_blueprint',
+ 'slug',
+ 'title',
+ 'type',
+ 'blocks_json',
+ 'status',
+ 'order',
+ 'created_at',
+ 'updated_at',
+ ]
+ read_only_fields = [
+ 'site_blueprint',
+ 'created_at',
+ 'updated_at',
+ ]
+
+
+class SiteBlueprintSerializer(serializers.ModelSerializer):
+ pages = PageBlueprintSerializer(many=True, read_only=True)
+ site_id = serializers.IntegerField(write_only=True, required=False)
+ sector_id = serializers.IntegerField(write_only=True, required=False)
+
+ class Meta:
+ model = SiteBlueprint
+ fields = [
+ 'id',
+ 'name',
+ 'description',
+ 'config_json',
+ 'structure_json',
+ 'status',
+ 'hosting_type',
+ 'version',
+ 'deployed_version',
+ 'site_id',
+ 'sector_id',
+ 'created_at',
+ 'updated_at',
+ 'pages',
+ ]
+ read_only_fields = [
+ 'structure_json',
+ 'status',
+ 'created_at',
+ 'updated_at',
+ 'pages',
+ ]
+
+ def validate(self, attrs):
+ site_id = attrs.pop('site_id', None)
+ sector_id = attrs.pop('sector_id', None)
+ if self.instance is None:
+ if not site_id:
+ raise serializers.ValidationError({'site_id': 'This field is required.'})
+ if not sector_id:
+ raise serializers.ValidationError({'sector_id': 'This field is required.'})
+ attrs['site_id'] = site_id
+ attrs['sector_id'] = sector_id
+ return attrs
+
diff --git a/backend/igny8_core/modules/site_builder/urls.py b/backend/igny8_core/modules/site_builder/urls.py
new file mode 100644
index 00000000..8487babc
--- /dev/null
+++ b/backend/igny8_core/modules/site_builder/urls.py
@@ -0,0 +1,18 @@
+from django.urls import include, path
+from rest_framework.routers import DefaultRouter
+
+from igny8_core.modules.site_builder.views import (
+ PageBlueprintViewSet,
+ SiteAssetView,
+ SiteBlueprintViewSet,
+)
+
+router = DefaultRouter()
+router.register(r'blueprints', SiteBlueprintViewSet, basename='site_blueprint')
+router.register(r'pages', PageBlueprintViewSet, basename='page_blueprint')
+
+urlpatterns = [
+ path('', include(router.urls)),
+ path('assets/', SiteAssetView.as_view(), name='site_builder_assets'),
+]
+
diff --git a/backend/igny8_core/modules/site_builder/views.py b/backend/igny8_core/modules/site_builder/views.py
new file mode 100644
index 00000000..2c66f5f7
--- /dev/null
+++ b/backend/igny8_core/modules/site_builder/views.py
@@ -0,0 +1,150 @@
+from rest_framework import status
+from rest_framework.decorators import action
+from rest_framework.permissions import IsAuthenticated
+from rest_framework.response import Response
+from rest_framework.views import APIView
+from rest_framework.exceptions import ValidationError
+
+from igny8_core.api.base import SiteSectorModelViewSet
+from igny8_core.api.permissions import IsAuthenticatedAndActive, IsEditorOrAbove
+from igny8_core.api.response import success_response, error_response
+from igny8_core.api.throttles import DebugScopedRateThrottle
+from igny8_core.business.site_building.models import SiteBlueprint, PageBlueprint
+from igny8_core.business.site_building.services import (
+ PageGenerationService,
+ SiteBuilderFileService,
+ StructureGenerationService,
+)
+from igny8_core.modules.site_builder.serializers import (
+ PageBlueprintSerializer,
+ SiteBlueprintSerializer,
+)
+
+
+class SiteBlueprintViewSet(SiteSectorModelViewSet):
+ """
+ CRUD + AI actions for site blueprints.
+ """
+
+ queryset = SiteBlueprint.objects.all().prefetch_related('pages')
+ serializer_class = SiteBlueprintSerializer
+ permission_classes = [IsAuthenticatedAndActive, IsEditorOrAbove]
+ throttle_scope = 'site_builder'
+ throttle_classes = [DebugScopedRateThrottle]
+
+ def perform_create(self, serializer):
+ from igny8_core.auth.models import Site, Sector
+
+ site_id = serializer.validated_data.pop('site_id', None)
+ sector_id = serializer.validated_data.pop('sector_id', None)
+ if not site_id or not sector_id:
+ raise ValidationError({'detail': 'site_id and sector_id are required.'})
+
+ try:
+ site = Site.objects.get(id=site_id)
+ except Site.DoesNotExist:
+ raise ValidationError({'site_id': 'Site not found.'})
+
+ try:
+ sector = Sector.objects.get(id=sector_id, site=site)
+ except Sector.DoesNotExist:
+ raise ValidationError({'sector_id': 'Sector does not belong to the selected site.'})
+
+ serializer.save(account=site.account, site=site, sector=sector)
+
+ @action(detail=True, methods=['post'])
+ def generate_structure(self, request, pk=None):
+ blueprint = self.get_object()
+ business_brief = request.data.get('business_brief') or \
+ blueprint.config_json.get('business_brief', '')
+ objectives = request.data.get('objectives') or \
+ blueprint.config_json.get('objectives', [])
+ style = request.data.get('style') or \
+ blueprint.config_json.get('style', {})
+
+ service = StructureGenerationService()
+ result = service.generate_structure(
+ site_blueprint=blueprint,
+ business_brief=business_brief,
+ objectives=objectives,
+ style_preferences=style,
+ metadata=request.data.get('metadata', {}),
+ )
+ return Response(result, status=status.HTTP_202_ACCEPTED if 'task_id' in result else status.HTTP_200_OK)
+
+
+class PageBlueprintViewSet(SiteSectorModelViewSet):
+ """
+ CRUD endpoints for page blueprints with content generation hooks.
+ """
+
+ queryset = PageBlueprint.objects.select_related('site_blueprint')
+ serializer_class = PageBlueprintSerializer
+ permission_classes = [IsAuthenticatedAndActive, IsEditorOrAbove]
+ throttle_scope = 'site_builder'
+ throttle_classes = [DebugScopedRateThrottle]
+
+ def perform_create(self, serializer):
+ page = serializer.save()
+ # Align account/site/sector with parent blueprint
+ page.account = page.site_blueprint.account
+ page.site = page.site_blueprint.site
+ page.sector = page.site_blueprint.sector
+ page.save(update_fields=['account', 'site', 'sector'])
+
+ @action(detail=True, methods=['post'])
+ def generate_content(self, request, pk=None):
+ page = self.get_object()
+ service = PageGenerationService()
+ result = service.generate_page_content(page, force_regenerate=request.data.get('force', False))
+ return success_response(result, request=request)
+
+ @action(detail=True, methods=['post'])
+ def regenerate(self, request, pk=None):
+ page = self.get_object()
+ service = PageGenerationService()
+ result = service.regenerate_page(page)
+ return success_response(result, request=request)
+
+
+class SiteAssetView(APIView):
+ """
+ File management for Site Builder assets.
+ """
+
+ permission_classes = [IsAuthenticated]
+
+ def __init__(self, **kwargs):
+ super().__init__(**kwargs)
+ self.file_service = SiteBuilderFileService()
+
+ def get(self, request, *args, **kwargs):
+ site_id = request.query_params.get('site_id')
+ folder = request.query_params.get('folder')
+ if not site_id:
+ return error_response('site_id is required', status.HTTP_400_BAD_REQUEST, request)
+ files = self.file_service.list_files(request.user, int(site_id), folder=folder)
+ return success_response({'files': files}, request)
+
+ def post(self, request, *args, **kwargs):
+ site_id = request.data.get('site_id')
+ version = int(request.data.get('version', 1))
+ folder = request.data.get('folder', 'images')
+ upload = request.FILES.get('file')
+ if not site_id or not upload:
+ return error_response('site_id and file are required', status.HTTP_400_BAD_REQUEST, request)
+ info = self.file_service.upload_file(request.user, int(site_id), upload, folder=folder, version=version)
+ return success_response(info, request, status.HTTP_201_CREATED)
+
+ def delete(self, request, *args, **kwargs):
+ site_id = request.data.get('site_id')
+ file_path = request.data.get('path')
+ version = int(request.data.get('version', 1))
+ if not site_id or not file_path:
+ return error_response('site_id and path are required', status.HTTP_400_BAD_REQUEST, request)
+ deleted = self.file_service.delete_file(request.user, int(site_id), file_path, version=version)
+ if deleted:
+ return success_response({'deleted': True}, request, status.HTTP_204_NO_CONTENT)
+ return error_response('File not found', status.HTTP_404_NOT_FOUND, request)
+
+
diff --git a/backend/igny8_core/settings.py b/backend/igny8_core/settings.py
index b2fd8df8..75a9fdc6 100644
--- a/backend/igny8_core/settings.py
+++ b/backend/igny8_core/settings.py
@@ -52,6 +52,8 @@ INSTALLED_APPS = [
'igny8_core.modules.system.apps.SystemConfig',
'igny8_core.modules.billing.apps.BillingConfig',
'igny8_core.modules.automation.apps.AutomationConfig',
+ 'igny8_core.business.site_building.apps.SiteBuildingConfig',
+ 'igny8_core.modules.site_builder.apps.SiteBuilderConfig',
]
# System module needs explicit registration for admin
diff --git a/backend/igny8_core/urls.py b/backend/igny8_core/urls.py
index 56687169..7ea5bffa 100644
--- a/backend/igny8_core/urls.py
+++ b/backend/igny8_core/urls.py
@@ -27,6 +27,7 @@ urlpatterns = [
path('api/v1/auth/', include('igny8_core.auth.urls')), # Auth endpoints
path('api/v1/planner/', include('igny8_core.modules.planner.urls')),
path('api/v1/writer/', include('igny8_core.modules.writer.urls')),
+ path('api/v1/site-builder/', include('igny8_core.modules.site_builder.urls')),
path('api/v1/system/', include('igny8_core.modules.system.urls')),
path('api/v1/billing/', include('igny8_core.modules.billing.urls')), # Billing endpoints
path('api/v1/automation/', include('igny8_core.modules.automation.urls')), # Automation endpoints
diff --git a/backend/test_image_write_access.py b/backend/test_image_write_access.py
deleted file mode 100644
index 427c2c8d..00000000
--- a/backend/test_image_write_access.py
+++ /dev/null
@@ -1,126 +0,0 @@
-"""
-Test script to verify write access to image directories
-"""
-import os
-import sys
-from pathlib import Path
-
-# Add project to path
-sys.path.insert(0, str(Path(__file__).parent))
-
-# Test write access logic
-def test_write_access():
- print("=" * 60)
- print("Testing Image Directory Write Access")
- print("=" * 60)
-
- # Test 1: Absolute path /data/app/images
- images_dir = '/data/app/images'
- write_test_passed = False
-
- print(f"\n[Test 1] Testing absolute path: {images_dir}")
- try:
- os.makedirs(images_dir, exist_ok=True)
- print(f" ✓ Directory created/verified: {images_dir}")
-
- # Test write access
- test_file = os.path.join(images_dir, '.write_test')
- print(f" → Attempting to write test file: {test_file}")
-
- with open(test_file, 'w') as f:
- f.write('test')
- print(f" ✓ Write successful")
-
- os.remove(test_file)
- print(f" ✓ Test file removed")
-
- write_test_passed = True
- print(f" ✅ SUCCESS: {images_dir} is writable")
-
- except PermissionError as e:
- print(f" ✗ PERMISSION DENIED: {e}")
- print(f" → Trying fallback path...")
-
- # Fallback to project-relative path
- try:
- from django.conf import settings
- base_dir = Path(settings.BASE_DIR) if hasattr(settings, 'BASE_DIR') else Path(__file__).resolve().parent.parent
- except:
- base_dir = Path(__file__).resolve().parent
-
- images_dir = str(base_dir / 'data' / 'app' / 'images')
- print(f"\n[Test 2] Testing fallback path: {images_dir}")
-
- try:
- os.makedirs(images_dir, exist_ok=True)
- print(f" ✓ Directory created/verified: {images_dir}")
-
- # Test fallback directory write access
- test_file = os.path.join(images_dir, '.write_test')
- print(f" → Attempting to write test file: {test_file}")
-
- with open(test_file, 'w') as f:
- f.write('test')
- print(f" ✓ Write successful")
-
- os.remove(test_file)
- print(f" ✓ Test file removed")
-
- write_test_passed = True
- print(f" ✅ SUCCESS: {images_dir} is writable")
-
- except Exception as fallback_error:
- print(f" ✗ FAILED: {fallback_error}")
- print(f" ❌ ERROR: Neither /data/app/images nor {images_dir} is writable")
- return False
-
- except Exception as e:
- print(f" ✗ ERROR: {e}")
- print(f" → Trying fallback path...")
-
- # Fallback to project-relative path
- try:
- from django.conf import settings
- base_dir = Path(settings.BASE_DIR) if hasattr(settings, 'BASE_DIR') else Path(__file__).resolve().parent.parent
- except:
- base_dir = Path(__file__).resolve().parent
-
- images_dir = str(base_dir / 'data' / 'app' / 'images')
- print(f"\n[Test 2] Testing fallback path: {images_dir}")
-
- try:
- os.makedirs(images_dir, exist_ok=True)
- print(f" ✓ Directory created/verified: {images_dir}")
-
- # Test fallback directory write access
- test_file = os.path.join(images_dir, '.write_test')
- print(f" → Attempting to write test file: {test_file}")
-
- with open(test_file, 'w') as f:
- f.write('test')
- print(f" ✓ Write successful")
-
- os.remove(test_file)
- print(f" ✓ Test file removed")
-
- write_test_passed = True
- print(f" ✅ SUCCESS: {images_dir} is writable")
-
- except Exception as fallback_error:
- print(f" ✗ FAILED: {fallback_error}")
- print(f" ❌ ERROR: Neither /data/app/images nor {images_dir} is writable")
- return False
-
- if not write_test_passed:
- print(f"\n❌ FAILED: No writable directory found")
- return False
-
- print(f"\n" + "=" * 60)
- print(f"✅ FINAL RESULT: Images will be saved to: {images_dir}")
- print("=" * 60)
- return True
-
-if __name__ == '__main__':
- success = test_write_access()
- sys.exit(0 if success else 1)
-
diff --git a/backend/update_free_plan.py b/backend/update_free_plan.py
deleted file mode 100644
index 11609690..00000000
--- a/backend/update_free_plan.py
+++ /dev/null
@@ -1,60 +0,0 @@
-#!/usr/bin/env python
-"""
-Script to fix and update the Free plan with proper values.
-"""
-import os
-import django
-import sys
-
-# Setup Django
-os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'igny8_core.settings')
-django.setup()
-
-from igny8_core.auth.models import Plan
-from decimal import Decimal
-
-def update_free_plan():
- """Update the Free plan with proper values and fix JSON fields."""
- try:
- free_plan = Plan.objects.get(slug='free')
-
- print(f"Found Free plan: {free_plan.name}")
- print("Updating values...")
-
- # Update free plan values - keeping existing values but fixing JSON fields
- # Fix JSON fields - use proper Python lists/dicts
- free_plan.image_model_choices = [] # Empty list for free plan (no image generation)
-
- # Ensure features is a proper list (not dict)
- if isinstance(free_plan.features, dict):
- free_plan.features = []
- elif not isinstance(free_plan.features, list):
- free_plan.features = []
-
- # Save the plan
- free_plan.save()
-
- print("✅ Successfully updated Free plan!")
- print(f" - Image Model Choices: {free_plan.image_model_choices}")
- print(f" - Features: {free_plan.features}")
- print(f" - Max Sites: {free_plan.max_sites}")
- print(f" - Max Users: {free_plan.max_users}")
- print(f" - Max Keywords: {free_plan.max_keywords}")
- print(f" - Max Clusters: {free_plan.max_clusters}")
- print(f" - Max Content Ideas: {free_plan.max_content_ideas}")
- print(f" - Monthly AI Credits: {free_plan.monthly_ai_credit_limit}")
- print(f" - Daily AI Request Limit: {free_plan.daily_ai_request_limit}")
- print(f" - Daily Image Generation Limit: {free_plan.daily_image_generation_limit}")
-
- except Plan.DoesNotExist:
- print("❌ Free plan not found!")
- sys.exit(1)
- except Exception as e:
- print(f"❌ Error updating free plan: {e}")
- import traceback
- traceback.print_exc()
- sys.exit(1)
-
-if __name__ == '__main__':
- update_free_plan()
-
diff --git a/docker-compose.app.yml b/docker-compose.app.yml
index 35490b85..e1963d3e 100644
--- a/docker-compose.app.yml
+++ b/docker-compose.app.yml
@@ -100,6 +100,25 @@ services:
- "com.docker.compose.project=igny8-app"
- "com.docker.compose.service=igny8_marketing_dev"
+ igny8_site_builder:
+ image: igny8-site-builder-dev:latest
+ container_name: igny8_site_builder
+ restart: always
+ ports:
+ - "0.0.0.0:8025:5175"
+ environment:
+ VITE_API_URL: "https://api.igny8.com/api"
+ volumes:
+ - /data/app/igny8/site-builder:/app:rw
+ - /data/app/igny8/frontend:/frontend:ro
+ depends_on:
+ igny8_backend:
+ condition: service_healthy
+ networks: [igny8_net]
+ labels:
+ - "com.docker.compose.project=igny8-app"
+ - "com.docker.compose.service=igny8_site_builder"
+
igny8_celery_worker:
image: igny8-backend:latest
container_name: igny8_celery_worker
diff --git a/frontend/src/components/shared/blocks/FeatureGridBlock.tsx b/frontend/src/components/shared/blocks/FeatureGridBlock.tsx
new file mode 100644
index 00000000..ca639b52
--- /dev/null
+++ b/frontend/src/components/shared/blocks/FeatureGridBlock.tsx
@@ -0,0 +1,30 @@
+import './blocks.css';
+
+export interface FeatureGridBlockProps {
+ heading?: string;
+ features: Array<{
+ title: string;
+ description?: string;
+ icon?: string;
+ }>;
+ columns?: 2 | 3 | 4;
+}
+
+export function FeatureGridBlock({ heading, features, columns = 3 }: FeatureGridBlockProps) {
+ return (
+
+ {heading && {heading}
}
+
+ {features.map((feature) => (
+
+ {feature.icon && {feature.icon}}
+ {feature.title}
+ {feature.description && {feature.description}
}
+
+ ))}
+
+
+ );
+}
+
+
diff --git a/frontend/src/components/shared/blocks/HeroBlock.tsx b/frontend/src/components/shared/blocks/HeroBlock.tsx
new file mode 100644
index 00000000..68e148ec
--- /dev/null
+++ b/frontend/src/components/shared/blocks/HeroBlock.tsx
@@ -0,0 +1,29 @@
+import type { ReactNode } from 'react';
+import './blocks.css';
+
+export interface HeroBlockProps {
+ eyebrow?: string;
+ title: string;
+ subtitle?: string;
+ ctaLabel?: string;
+ onCtaClick?: () => void;
+ supportingContent?: ReactNode;
+}
+
+export function HeroBlock({ eyebrow, title, subtitle, ctaLabel, onCtaClick, supportingContent }: HeroBlockProps) {
+ return (
+
+ {eyebrow && {eyebrow}
}
+ {title}
+ {subtitle && {subtitle}
}
+ {supportingContent && {supportingContent}
}
+ {ctaLabel && (
+
+ )}
+
+ );
+}
+
+
diff --git a/frontend/src/components/shared/blocks/StatsPanel.tsx b/frontend/src/components/shared/blocks/StatsPanel.tsx
new file mode 100644
index 00000000..074c4363
--- /dev/null
+++ b/frontend/src/components/shared/blocks/StatsPanel.tsx
@@ -0,0 +1,31 @@
+import './blocks.css';
+
+export interface StatItem {
+ label: string;
+ value: string;
+ description?: string;
+}
+
+export interface StatsPanelProps {
+ heading?: string;
+ stats: StatItem[];
+}
+
+export function StatsPanel({ heading, stats }: StatsPanelProps) {
+ return (
+
+ {heading && {heading}
}
+
+ {stats.map((stat) => (
+
+
{stat.value}
+
{stat.label}
+ {stat.description &&
{stat.description}
}
+
+ ))}
+
+
+ );
+}
+
+
diff --git a/frontend/src/components/shared/blocks/blocks.css b/frontend/src/components/shared/blocks/blocks.css
new file mode 100644
index 00000000..661ad9a8
--- /dev/null
+++ b/frontend/src/components/shared/blocks/blocks.css
@@ -0,0 +1,123 @@
+.shared-hero {
+ padding: 2.5rem;
+ border-radius: 24px;
+ background: linear-gradient(135deg, #4c1d95, #312e81);
+ color: #fff;
+ display: flex;
+ flex-direction: column;
+ gap: 1rem;
+}
+
+.shared-hero__eyebrow {
+ text-transform: uppercase;
+ letter-spacing: 0.12em;
+ font-size: 0.75rem;
+ margin: 0;
+ opacity: 0.8;
+}
+
+.shared-hero__title {
+ margin: 0;
+ font-size: clamp(2rem, 5vw, 3rem);
+ font-weight: 700;
+}
+
+.shared-hero__subtitle {
+ margin: 0;
+ color: rgba(255, 255, 255, 0.85);
+ font-size: 1.05rem;
+}
+
+.shared-button {
+ border: none;
+ border-radius: 999px;
+ padding: 0.9rem 1.8rem;
+ font-size: 0.95rem;
+ font-weight: 600;
+ color: #0f172a;
+ background: #fff;
+ align-self: flex-start;
+ cursor: pointer;
+}
+
+.shared-card {
+ padding: 2rem;
+ border-radius: 22px;
+ background: #fff;
+ box-shadow: 0 18px 45px rgba(15, 23, 42, 0.08);
+ display: flex;
+ flex-direction: column;
+ gap: 1.25rem;
+}
+
+.shared-card h3 {
+ margin: 0;
+ font-size: 1.35rem;
+ color: #0f172a;
+}
+
+.shared-grid {
+ display: grid;
+ gap: 1rem;
+}
+
+.shared-grid--2 {
+ grid-template-columns: repeat(2, minmax(0, 1fr));
+}
+
+.shared-grid--3 {
+ grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
+}
+
+.shared-grid--4 {
+ grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
+}
+
+.shared-feature {
+ padding: 1rem;
+ border-radius: 16px;
+ border: 1px solid rgba(15, 23, 42, 0.08);
+ background: #f8fafc;
+}
+
+.shared-feature h4 {
+ margin: 0;
+ font-size: 1.05rem;
+ color: #0f172a;
+}
+
+.shared-feature p {
+ margin: 0.5rem 0 0;
+ color: #475569;
+}
+
+.shared-stats {
+ display: grid;
+ grid-template-columns: repeat(auto-fit, minmax(160px, 1fr));
+ gap: 0.75rem;
+}
+
+.shared-stat {
+ padding: 1rem;
+ border-radius: 16px;
+ background: rgba(15, 23, 42, 0.03);
+}
+
+.shared-stat strong {
+ display: block;
+ font-size: 1.6rem;
+ color: #0f172a;
+}
+
+.shared-stat span {
+ display: block;
+ color: #475569;
+}
+
+.shared-stat p {
+ margin: 0.35rem 0 0;
+ color: #64748b;
+ font-size: 0.9rem;
+}
+
+
diff --git a/frontend/src/components/shared/blocks/index.ts b/frontend/src/components/shared/blocks/index.ts
new file mode 100644
index 00000000..15e646d5
--- /dev/null
+++ b/frontend/src/components/shared/blocks/index.ts
@@ -0,0 +1,8 @@
+export { HeroBlock } from './HeroBlock';
+export type { HeroBlockProps } from './HeroBlock';
+export { FeatureGridBlock } from './FeatureGridBlock';
+export type { FeatureGridBlockProps } from './FeatureGridBlock';
+export { StatsPanel } from './StatsPanel';
+export type { StatsPanelProps, StatItem } from './StatsPanel';
+
+
diff --git a/frontend/src/components/shared/index.ts b/frontend/src/components/shared/index.ts
new file mode 100644
index 00000000..86cc514b
--- /dev/null
+++ b/frontend/src/components/shared/index.ts
@@ -0,0 +1,9 @@
+export * from './blocks';
+export * from './layouts';
+export * from './templates';
+
+export * from './blocks';
+export * from './layouts';
+export * from './templates';
+
+
diff --git a/frontend/src/components/shared/layouts/DefaultLayout.tsx b/frontend/src/components/shared/layouts/DefaultLayout.tsx
new file mode 100644
index 00000000..2260d152
--- /dev/null
+++ b/frontend/src/components/shared/layouts/DefaultLayout.tsx
@@ -0,0 +1,28 @@
+import type { ReactNode } from 'react';
+import './layouts.css';
+
+export interface DefaultLayoutProps {
+ hero?: ReactNode;
+ sections: ReactNode[];
+ sidebar?: ReactNode;
+}
+
+export function DefaultLayout({ hero, sections, sidebar }: DefaultLayoutProps) {
+ return (
+
+ {hero &&
{hero}
}
+
+
+ {sections.map((section, index) => (
+
+ {section}
+
+ ))}
+
+ {sidebar &&
}
+
+
+ );
+}
+
+
diff --git a/frontend/src/components/shared/layouts/MinimalLayout.tsx b/frontend/src/components/shared/layouts/MinimalLayout.tsx
new file mode 100644
index 00000000..b629ee07
--- /dev/null
+++ b/frontend/src/components/shared/layouts/MinimalLayout.tsx
@@ -0,0 +1,13 @@
+import type { ReactNode } from 'react';
+import './layouts.css';
+
+export interface MinimalLayoutProps {
+ children: ReactNode;
+ background?: 'light' | 'dark';
+}
+
+export function MinimalLayout({ children, background = 'light' }: MinimalLayoutProps) {
+ return {children}
;
+}
+
+
diff --git a/frontend/src/components/shared/layouts/index.ts b/frontend/src/components/shared/layouts/index.ts
new file mode 100644
index 00000000..fa68cd89
--- /dev/null
+++ b/frontend/src/components/shared/layouts/index.ts
@@ -0,0 +1,6 @@
+export { DefaultLayout } from './DefaultLayout';
+export type { DefaultLayoutProps } from './DefaultLayout';
+export { MinimalLayout } from './MinimalLayout';
+export type { MinimalLayoutProps } from './MinimalLayout';
+
+
diff --git a/frontend/src/components/shared/layouts/layouts.css b/frontend/src/components/shared/layouts/layouts.css
new file mode 100644
index 00000000..2ee5b190
--- /dev/null
+++ b/frontend/src/components/shared/layouts/layouts.css
@@ -0,0 +1,65 @@
+.shared-layout {
+ display: flex;
+ flex-direction: column;
+ gap: 1.5rem;
+}
+
+.shared-layout__hero {
+ min-height: 320px;
+}
+
+.shared-layout__body {
+ display: grid;
+ grid-template-columns: minmax(0, 1fr);
+ gap: 1.25rem;
+}
+
+@media (min-width: 1024px) {
+ .shared-layout__body {
+ grid-template-columns: minmax(0, 2.3fr) minmax(0, 1fr);
+ }
+}
+
+.shared-layout__section {
+ margin-bottom: 1.25rem;
+}
+
+.shared-layout__sidebar {
+ position: sticky;
+ top: 3rem;
+ align-self: flex-start;
+ background: #0f172a;
+ color: #fff;
+ border-radius: 24px;
+ padding: 1.75rem;
+ box-shadow: 0 20px 45px rgba(15, 23, 42, 0.35);
+}
+
+.shared-layout__minimal {
+ border-radius: 30px;
+ padding: 2.5rem;
+}
+
+.shared-layout__minimal--light {
+ background: #f8fafc;
+ border: 1px solid rgba(15, 23, 42, 0.06);
+}
+
+.shared-layout__minimal--dark {
+ background: #0f172a;
+ color: #fff;
+}
+
+.shared-landing {
+ display: flex;
+ flex-direction: column;
+ gap: 1.5rem;
+}
+
+.shared-landing__highlights {
+ display: grid;
+ grid-template-columns: repeat(auto-fit, minmax(220px, 1fr));
+ gap: 1rem;
+}
+
+
diff --git a/frontend/src/components/shared/templates/LandingTemplate.tsx b/frontend/src/components/shared/templates/LandingTemplate.tsx
new file mode 100644
index 00000000..2ba9e16c
--- /dev/null
+++ b/frontend/src/components/shared/templates/LandingTemplate.tsx
@@ -0,0 +1,26 @@
+import type { ReactNode } from 'react';
+import { MinimalLayout } from '../layouts/MinimalLayout';
+
+export interface LandingTemplateProps {
+ hero: ReactNode;
+ highlights: ReactNode[];
+ background?: 'light' | 'dark';
+}
+
+export function LandingTemplate({ hero, highlights, background }: LandingTemplateProps) {
+ return (
+
+
+
{hero}
+
+ {highlights.map((highlight, index) => (
+ // eslint-disable-next-line react/no-array-index-key
+
{highlight}
+ ))}
+
+
+
+ );
+}
+
+
diff --git a/frontend/src/components/shared/templates/MarketingTemplate.tsx b/frontend/src/components/shared/templates/MarketingTemplate.tsx
new file mode 100644
index 00000000..83e7d134
--- /dev/null
+++ b/frontend/src/components/shared/templates/MarketingTemplate.tsx
@@ -0,0 +1,14 @@
+import type { ReactNode } from 'react';
+import { DefaultLayout } from '../layouts/DefaultLayout';
+
+export interface MarketingTemplateProps {
+ hero: ReactNode;
+ sections: ReactNode[];
+ sidebar?: ReactNode;
+}
+
+export function MarketingTemplate(props: MarketingTemplateProps) {
+ return ;
+}
+
+
diff --git a/frontend/src/components/shared/templates/index.ts b/frontend/src/components/shared/templates/index.ts
new file mode 100644
index 00000000..ce61846b
--- /dev/null
+++ b/frontend/src/components/shared/templates/index.ts
@@ -0,0 +1,6 @@
+export { MarketingTemplate } from './MarketingTemplate';
+export type { MarketingTemplateProps } from './MarketingTemplate';
+export { LandingTemplate } from './LandingTemplate';
+export type { LandingTemplateProps } from './LandingTemplate';
+
+
diff --git a/site-builder/.gitignore b/site-builder/.gitignore
new file mode 100644
index 00000000..a547bf36
--- /dev/null
+++ b/site-builder/.gitignore
@@ -0,0 +1,24 @@
+# Logs
+logs
+*.log
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+pnpm-debug.log*
+lerna-debug.log*
+
+node_modules
+dist
+dist-ssr
+*.local
+
+# Editor directories and files
+.vscode/*
+!.vscode/extensions.json
+.idea
+.DS_Store
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
+*.sw?
diff --git a/site-builder/Dockerfile.dev b/site-builder/Dockerfile.dev
new file mode 100644
index 00000000..27e54f04
--- /dev/null
+++ b/site-builder/Dockerfile.dev
@@ -0,0 +1,18 @@
+# Site Builder Dev Image (Node 22 to satisfy Vite requirements)
+FROM node:22-alpine
+
+WORKDIR /app
+
+# Copy package manifests first for better caching
+COPY package*.json ./
+
+RUN npm install
+
+# Copy source (still bind-mounted at runtime, but needed for initial run)
+COPY . .
+
+EXPOSE 5175
+
+CMD ["npm", "run", "dev", "--", "--host", "0.0.0.0", "--port", "5175"]
+
+
diff --git a/site-builder/README.md b/site-builder/README.md
new file mode 100644
index 00000000..d2e77611
--- /dev/null
+++ b/site-builder/README.md
@@ -0,0 +1,73 @@
+# React + TypeScript + Vite
+
+This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules.
+
+Currently, two official plugins are available:
+
+- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react) uses [Babel](https://babeljs.io/) (or [oxc](https://oxc.rs) when used in [rolldown-vite](https://vite.dev/guide/rolldown)) for Fast Refresh
+- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh
+
+## React Compiler
+
+The React Compiler is not enabled on this template because of its impact on dev & build performances. To add it, see [this documentation](https://react.dev/learn/react-compiler/installation).
+
+## Expanding the ESLint configuration
+
+If you are developing a production application, we recommend updating the configuration to enable type-aware lint rules:
+
+```js
+export default defineConfig([
+ globalIgnores(['dist']),
+ {
+ files: ['**/*.{ts,tsx}'],
+ extends: [
+ // Other configs...
+
+ // Remove tseslint.configs.recommended and replace with this
+ tseslint.configs.recommendedTypeChecked,
+ // Alternatively, use this for stricter rules
+ tseslint.configs.strictTypeChecked,
+ // Optionally, add this for stylistic rules
+ tseslint.configs.stylisticTypeChecked,
+
+ // Other configs...
+ ],
+ languageOptions: {
+ parserOptions: {
+ project: ['./tsconfig.node.json', './tsconfig.app.json'],
+ tsconfigRootDir: import.meta.dirname,
+ },
+ // other options...
+ },
+ },
+])
+```
+
+You can also install [eslint-plugin-react-x](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-x) and [eslint-plugin-react-dom](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-dom) for React-specific lint rules:
+
+```js
+// eslint.config.js
+import reactX from 'eslint-plugin-react-x'
+import reactDom from 'eslint-plugin-react-dom'
+
+export default defineConfig([
+ globalIgnores(['dist']),
+ {
+ files: ['**/*.{ts,tsx}'],
+ extends: [
+ // Other configs...
+ // Enable lint rules for React
+ reactX.configs['recommended-typescript'],
+ // Enable lint rules for React DOM
+ reactDom.configs.recommended,
+ ],
+ languageOptions: {
+ parserOptions: {
+ project: ['./tsconfig.node.json', './tsconfig.app.json'],
+ tsconfigRootDir: import.meta.dirname,
+ },
+ // other options...
+ },
+ },
+])
+```
diff --git a/site-builder/eslint.config.js b/site-builder/eslint.config.js
new file mode 100644
index 00000000..5e6b472f
--- /dev/null
+++ b/site-builder/eslint.config.js
@@ -0,0 +1,23 @@
+import js from '@eslint/js'
+import globals from 'globals'
+import reactHooks from 'eslint-plugin-react-hooks'
+import reactRefresh from 'eslint-plugin-react-refresh'
+import tseslint from 'typescript-eslint'
+import { defineConfig, globalIgnores } from 'eslint/config'
+
+export default defineConfig([
+ globalIgnores(['dist']),
+ {
+ files: ['**/*.{ts,tsx}'],
+ extends: [
+ js.configs.recommended,
+ tseslint.configs.recommended,
+ reactHooks.configs.flat.recommended,
+ reactRefresh.configs.vite,
+ ],
+ languageOptions: {
+ ecmaVersion: 2020,
+ globals: globals.browser,
+ },
+ },
+])
diff --git a/site-builder/index.html b/site-builder/index.html
new file mode 100644
index 00000000..fe1cf335
--- /dev/null
+++ b/site-builder/index.html
@@ -0,0 +1,13 @@
+
+
+
+
+
+
+ site-builder
+
+
+
+
+
+
diff --git a/site-builder/package-lock.json b/site-builder/package-lock.json
new file mode 100644
index 00000000..1e44862a
--- /dev/null
+++ b/site-builder/package-lock.json
@@ -0,0 +1,3894 @@
+{
+ "name": "site-builder",
+ "version": "0.0.0",
+ "lockfileVersion": 3,
+ "requires": true,
+ "packages": {
+ "": {
+ "name": "site-builder",
+ "version": "0.0.0",
+ "dependencies": {
+ "axios": "^1.13.2",
+ "lucide-react": "^0.554.0",
+ "react": "^19.2.0",
+ "react-dom": "^19.2.0",
+ "react-hook-form": "^7.66.0",
+ "react-router-dom": "^7.9.6",
+ "zustand": "^5.0.8"
+ },
+ "devDependencies": {
+ "@eslint/js": "^9.39.1",
+ "@types/node": "^24.10.0",
+ "@types/react": "^19.2.2",
+ "@types/react-dom": "^19.2.2",
+ "@types/react-router-dom": "^5.3.3",
+ "@vitejs/plugin-react": "^5.1.0",
+ "eslint": "^9.39.1",
+ "eslint-plugin-react-hooks": "^7.0.1",
+ "eslint-plugin-react-refresh": "^0.4.24",
+ "globals": "^16.5.0",
+ "typescript": "~5.9.3",
+ "typescript-eslint": "^8.46.3",
+ "vite": "^7.2.2"
+ }
+ },
+ "node_modules/@babel/code-frame": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz",
+ "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-validator-identifier": "^7.27.1",
+ "js-tokens": "^4.0.0",
+ "picocolors": "^1.1.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/compat-data": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.5.tgz",
+ "integrity": "sha512-6uFXyCayocRbqhZOB+6XcuZbkMNimwfVGFji8CTZnCzOHVGvDqzvitu1re2AU5LROliz7eQPhB8CpAMvnx9EjA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/core": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz",
+ "integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/code-frame": "^7.27.1",
+ "@babel/generator": "^7.28.5",
+ "@babel/helper-compilation-targets": "^7.27.2",
+ "@babel/helper-module-transforms": "^7.28.3",
+ "@babel/helpers": "^7.28.4",
+ "@babel/parser": "^7.28.5",
+ "@babel/template": "^7.27.2",
+ "@babel/traverse": "^7.28.5",
+ "@babel/types": "^7.28.5",
+ "@jridgewell/remapping": "^2.3.5",
+ "convert-source-map": "^2.0.0",
+ "debug": "^4.1.0",
+ "gensync": "^1.0.0-beta.2",
+ "json5": "^2.2.3",
+ "semver": "^6.3.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/babel"
+ }
+ },
+ "node_modules/@babel/generator": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz",
+ "integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/parser": "^7.28.5",
+ "@babel/types": "^7.28.5",
+ "@jridgewell/gen-mapping": "^0.3.12",
+ "@jridgewell/trace-mapping": "^0.3.28",
+ "jsesc": "^3.0.2"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-compilation-targets": {
+ "version": "7.27.2",
+ "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.2.tgz",
+ "integrity": "sha512-2+1thGUUWWjLTYTHZWK1n8Yga0ijBz1XAhUXcKy81rd5g6yh7hGqMp45v7cadSbEHc9G3OTv45SyneRN3ps4DQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/compat-data": "^7.27.2",
+ "@babel/helper-validator-option": "^7.27.1",
+ "browserslist": "^4.24.0",
+ "lru-cache": "^5.1.1",
+ "semver": "^6.3.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-globals": {
+ "version": "7.28.0",
+ "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz",
+ "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-module-imports": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.27.1.tgz",
+ "integrity": "sha512-0gSFWUPNXNopqtIPQvlD5WgXYI5GY2kP2cCvoT8kczjbfcfuIljTbcWrulD1CIPIX2gt1wghbDy08yE1p+/r3w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/traverse": "^7.27.1",
+ "@babel/types": "^7.27.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-module-transforms": {
+ "version": "7.28.3",
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.3.tgz",
+ "integrity": "sha512-gytXUbs8k2sXS9PnQptz5o0QnpLL51SwASIORY6XaBKF88nsOT0Zw9szLqlSGQDP/4TljBAD5y98p2U1fqkdsw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-module-imports": "^7.27.1",
+ "@babel/helper-validator-identifier": "^7.27.1",
+ "@babel/traverse": "^7.28.3"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0"
+ }
+ },
+ "node_modules/@babel/helper-plugin-utils": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.27.1.tgz",
+ "integrity": "sha512-1gn1Up5YXka3YYAHGKpbideQ5Yjf1tDa9qYcgysz+cNCXukyLl6DjPXhD3VRwSb8c0J9tA4b2+rHEZtc6R0tlw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-string-parser": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz",
+ "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-validator-identifier": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz",
+ "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-validator-option": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz",
+ "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helpers": {
+ "version": "7.28.4",
+ "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.4.tgz",
+ "integrity": "sha512-HFN59MmQXGHVyYadKLVumYsA9dBFun/ldYxipEjzA4196jpLZd8UjEEBLkbEkvfYreDqJhZxYAWFPtrfhNpj4w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/template": "^7.27.2",
+ "@babel/types": "^7.28.4"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/parser": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz",
+ "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/types": "^7.28.5"
+ },
+ "bin": {
+ "parser": "bin/babel-parser.js"
+ },
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/@babel/plugin-transform-react-jsx-self": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.27.1.tgz",
+ "integrity": "sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-plugin-utils": "^7.27.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0-0"
+ }
+ },
+ "node_modules/@babel/plugin-transform-react-jsx-source": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.27.1.tgz",
+ "integrity": "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-plugin-utils": "^7.27.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0-0"
+ }
+ },
+ "node_modules/@babel/template": {
+ "version": "7.27.2",
+ "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz",
+ "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/code-frame": "^7.27.1",
+ "@babel/parser": "^7.27.2",
+ "@babel/types": "^7.27.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/traverse": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz",
+ "integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/code-frame": "^7.27.1",
+ "@babel/generator": "^7.28.5",
+ "@babel/helper-globals": "^7.28.0",
+ "@babel/parser": "^7.28.5",
+ "@babel/template": "^7.27.2",
+ "@babel/types": "^7.28.5",
+ "debug": "^4.3.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/types": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz",
+ "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-string-parser": "^7.27.1",
+ "@babel/helper-validator-identifier": "^7.28.5"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@esbuild/aix-ppc64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz",
+ "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "aix"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/android-arm": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz",
+ "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/android-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz",
+ "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/android-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz",
+ "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/darwin-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz",
+ "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/darwin-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz",
+ "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/freebsd-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz",
+ "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/freebsd-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz",
+ "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-arm": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz",
+ "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz",
+ "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-ia32": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz",
+ "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-loong64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz",
+ "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==",
+ "cpu": [
+ "loong64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-mips64el": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz",
+ "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==",
+ "cpu": [
+ "mips64el"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-ppc64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz",
+ "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-riscv64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz",
+ "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-s390x": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz",
+ "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==",
+ "cpu": [
+ "s390x"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz",
+ "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/netbsd-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz",
+ "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "netbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/netbsd-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz",
+ "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "netbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/openbsd-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz",
+ "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "openbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/openbsd-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz",
+ "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "openbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/openharmony-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz",
+ "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "openharmony"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/sunos-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz",
+ "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "sunos"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/win32-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz",
+ "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/win32-ia32": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz",
+ "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/win32-x64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz",
+ "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@eslint-community/eslint-utils": {
+ "version": "4.9.0",
+ "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.0.tgz",
+ "integrity": "sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "eslint-visitor-keys": "^3.4.3"
+ },
+ "engines": {
+ "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0"
+ }
+ },
+ "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": {
+ "version": "3.4.3",
+ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz",
+ "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/@eslint-community/regexpp": {
+ "version": "4.12.2",
+ "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz",
+ "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^12.0.0 || ^14.0.0 || >=16.0.0"
+ }
+ },
+ "node_modules/@eslint/config-array": {
+ "version": "0.21.1",
+ "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.1.tgz",
+ "integrity": "sha512-aw1gNayWpdI/jSYVgzN5pL0cfzU02GT3NBpeT/DXbx1/1x7ZKxFPd9bwrzygx/qiwIQiJ1sw/zD8qY/kRvlGHA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@eslint/object-schema": "^2.1.7",
+ "debug": "^4.3.1",
+ "minimatch": "^3.1.2"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/config-helpers": {
+ "version": "0.4.2",
+ "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.4.2.tgz",
+ "integrity": "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@eslint/core": "^0.17.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/core": {
+ "version": "0.17.0",
+ "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.17.0.tgz",
+ "integrity": "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@types/json-schema": "^7.0.15"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/eslintrc": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.1.tgz",
+ "integrity": "sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ajv": "^6.12.4",
+ "debug": "^4.3.2",
+ "espree": "^10.0.1",
+ "globals": "^14.0.0",
+ "ignore": "^5.2.0",
+ "import-fresh": "^3.2.1",
+ "js-yaml": "^4.1.0",
+ "minimatch": "^3.1.2",
+ "strip-json-comments": "^3.1.1"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/@eslint/eslintrc/node_modules/globals": {
+ "version": "14.0.0",
+ "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz",
+ "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/@eslint/js": {
+ "version": "9.39.1",
+ "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.39.1.tgz",
+ "integrity": "sha512-S26Stp4zCy88tH94QbBv3XCuzRQiZ9yXofEILmglYTh/Ug/a9/umqvgFtYBAo3Lp0nsI/5/qH1CCrbdK3AP1Tw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://eslint.org/donate"
+ }
+ },
+ "node_modules/@eslint/object-schema": {
+ "version": "2.1.7",
+ "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.7.tgz",
+ "integrity": "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/plugin-kit": {
+ "version": "0.4.1",
+ "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.4.1.tgz",
+ "integrity": "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@eslint/core": "^0.17.0",
+ "levn": "^0.4.1"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@humanfs/core": {
+ "version": "0.19.1",
+ "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz",
+ "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=18.18.0"
+ }
+ },
+ "node_modules/@humanfs/node": {
+ "version": "0.16.7",
+ "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.7.tgz",
+ "integrity": "sha512-/zUx+yOsIrG4Y43Eh2peDeKCxlRt/gET6aHfaKpuq267qXdYDFViVHfMaLyygZOnl0kGWxFIgsBy8QFuTLUXEQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@humanfs/core": "^0.19.1",
+ "@humanwhocodes/retry": "^0.4.0"
+ },
+ "engines": {
+ "node": ">=18.18.0"
+ }
+ },
+ "node_modules/@humanwhocodes/module-importer": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz",
+ "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=12.22"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/nzakas"
+ }
+ },
+ "node_modules/@humanwhocodes/retry": {
+ "version": "0.4.3",
+ "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz",
+ "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=18.18"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/nzakas"
+ }
+ },
+ "node_modules/@jridgewell/gen-mapping": {
+ "version": "0.3.13",
+ "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
+ "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/sourcemap-codec": "^1.5.0",
+ "@jridgewell/trace-mapping": "^0.3.24"
+ }
+ },
+ "node_modules/@jridgewell/remapping": {
+ "version": "2.3.5",
+ "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz",
+ "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/gen-mapping": "^0.3.5",
+ "@jridgewell/trace-mapping": "^0.3.24"
+ }
+ },
+ "node_modules/@jridgewell/resolve-uri": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
+ "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/@jridgewell/sourcemap-codec": {
+ "version": "1.5.5",
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
+ "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@jridgewell/trace-mapping": {
+ "version": "0.3.31",
+ "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
+ "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/resolve-uri": "^3.1.0",
+ "@jridgewell/sourcemap-codec": "^1.4.14"
+ }
+ },
+ "node_modules/@nodelib/fs.scandir": {
+ "version": "2.1.5",
+ "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz",
+ "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@nodelib/fs.stat": "2.0.5",
+ "run-parallel": "^1.1.9"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/@nodelib/fs.stat": {
+ "version": "2.0.5",
+ "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz",
+ "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/@nodelib/fs.walk": {
+ "version": "1.2.8",
+ "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz",
+ "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@nodelib/fs.scandir": "2.1.5",
+ "fastq": "^1.6.0"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/@rolldown/pluginutils": {
+ "version": "1.0.0-beta.47",
+ "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.47.tgz",
+ "integrity": "sha512-8QagwMH3kNCuzD8EWL8R2YPW5e4OrHNSAHRFDdmFqEwEaD/KcNKjVoumo+gP2vW5eKB2UPbM6vTYiGZX0ixLnw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@rollup/rollup-android-arm-eabi": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.53.2.tgz",
+ "integrity": "sha512-yDPzwsgiFO26RJA4nZo8I+xqzh7sJTZIWQOxn+/XOdPE31lAvLIYCKqjV+lNH/vxE2L2iH3plKxDCRK6i+CwhA==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ]
+ },
+ "node_modules/@rollup/rollup-android-arm64": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.53.2.tgz",
+ "integrity": "sha512-k8FontTxIE7b0/OGKeSN5B6j25EuppBcWM33Z19JoVT7UTXFSo3D9CdU39wGTeb29NO3XxpMNauh09B+Ibw+9g==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ]
+ },
+ "node_modules/@rollup/rollup-darwin-arm64": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.53.2.tgz",
+ "integrity": "sha512-A6s4gJpomNBtJ2yioj8bflM2oogDwzUiMl2yNJ2v9E7++sHrSrsQ29fOfn5DM/iCzpWcebNYEdXpaK4tr2RhfQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ]
+ },
+ "node_modules/@rollup/rollup-darwin-x64": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.53.2.tgz",
+ "integrity": "sha512-e6XqVmXlHrBlG56obu9gDRPW3O3hLxpwHpLsBJvuI8qqnsrtSZ9ERoWUXtPOkY8c78WghyPHZdmPhHLWNdAGEw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ]
+ },
+ "node_modules/@rollup/rollup-freebsd-arm64": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.53.2.tgz",
+ "integrity": "sha512-v0E9lJW8VsrwPux5Qe5CwmH/CF/2mQs6xU1MF3nmUxmZUCHazCjLgYvToOk+YuuUqLQBio1qkkREhxhc656ViA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ]
+ },
+ "node_modules/@rollup/rollup-freebsd-x64": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.53.2.tgz",
+ "integrity": "sha512-ClAmAPx3ZCHtp6ysl4XEhWU69GUB1D+s7G9YjHGhIGCSrsg00nEGRRZHmINYxkdoJehde8VIsDC5t9C0gb6yqA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm-gnueabihf": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.53.2.tgz",
+ "integrity": "sha512-EPlb95nUsz6Dd9Qy13fI5kUPXNSljaG9FiJ4YUGU1O/Q77i5DYFW5KR8g1OzTcdZUqQQ1KdDqsTohdFVwCwjqg==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm-musleabihf": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.53.2.tgz",
+ "integrity": "sha512-BOmnVW+khAUX+YZvNfa0tGTEMVVEerOxN0pDk2E6N6DsEIa2Ctj48FOMfNDdrwinocKaC7YXUZ1pHlKpnkja/Q==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm64-gnu": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.53.2.tgz",
+ "integrity": "sha512-Xt2byDZ+6OVNuREgBXr4+CZDJtrVso5woFtpKdGPhpTPHcNG7D8YXeQzpNbFRxzTVqJf7kvPMCub/pcGUWgBjA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm64-musl": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.53.2.tgz",
+ "integrity": "sha512-+LdZSldy/I9N8+klim/Y1HsKbJ3BbInHav5qE9Iy77dtHC/pibw1SR/fXlWyAk0ThnpRKoODwnAuSjqxFRDHUQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-loong64-gnu": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.53.2.tgz",
+ "integrity": "sha512-8ms8sjmyc1jWJS6WdNSA23rEfdjWB30LH8Wqj0Cqvv7qSHnvw6kgMMXRdop6hkmGPlyYBdRPkjJnj3KCUHV/uQ==",
+ "cpu": [
+ "loong64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-ppc64-gnu": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.53.2.tgz",
+ "integrity": "sha512-3HRQLUQbpBDMmzoxPJYd3W6vrVHOo2cVW8RUo87Xz0JPJcBLBr5kZ1pGcQAhdZgX9VV7NbGNipah1omKKe23/g==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-riscv64-gnu": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.53.2.tgz",
+ "integrity": "sha512-fMjKi+ojnmIvhk34gZP94vjogXNNUKMEYs+EDaB/5TG/wUkoeua7p7VCHnE6T2Tx+iaghAqQX8teQzcvrYpaQA==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-riscv64-musl": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.53.2.tgz",
+ "integrity": "sha512-XuGFGU+VwUUV5kLvoAdi0Wz5Xbh2SrjIxCtZj6Wq8MDp4bflb/+ThZsVxokM7n0pcbkEr2h5/pzqzDYI7cCgLQ==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-s390x-gnu": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.53.2.tgz",
+ "integrity": "sha512-w6yjZF0P+NGzWR3AXWX9zc0DNEGdtvykB03uhonSHMRa+oWA6novflo2WaJr6JZakG2ucsyb+rvhrKac6NIy+w==",
+ "cpu": [
+ "s390x"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-x64-gnu": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.53.2.tgz",
+ "integrity": "sha512-yo8d6tdfdeBArzC7T/PnHd7OypfI9cbuZzPnzLJIyKYFhAQ8SvlkKtKBMbXDxe1h03Rcr7u++nFS7tqXz87Gtw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-x64-musl": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.53.2.tgz",
+ "integrity": "sha512-ah59c1YkCxKExPP8O9PwOvs+XRLKwh/mV+3YdKqQ5AMQ0r4M4ZDuOrpWkUaqO7fzAHdINzV9tEVu8vNw48z0lA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-openharmony-arm64": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.53.2.tgz",
+ "integrity": "sha512-4VEd19Wmhr+Zy7hbUsFZ6YXEiP48hE//KPLCSVNY5RMGX2/7HZ+QkN55a3atM1C/BZCGIgqN+xrVgtdak2S9+A==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "openharmony"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-arm64-msvc": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.53.2.tgz",
+ "integrity": "sha512-IlbHFYc/pQCgew/d5fslcy1KEaYVCJ44G8pajugd8VoOEI8ODhtb/j8XMhLpwHCMB3yk2J07ctup10gpw2nyMA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-ia32-msvc": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.53.2.tgz",
+ "integrity": "sha512-lNlPEGgdUfSzdCWU176ku/dQRnA7W+Gp8d+cWv73jYrb8uT7HTVVxq62DUYxjbaByuf1Yk0RIIAbDzp+CnOTFg==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-x64-gnu": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.53.2.tgz",
+ "integrity": "sha512-S6YojNVrHybQis2lYov1sd+uj7K0Q05NxHcGktuMMdIQ2VixGwAfbJ23NnlvvVV1bdpR2m5MsNBViHJKcA4ADw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-x64-msvc": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.53.2.tgz",
+ "integrity": "sha512-k+/Rkcyx//P6fetPoLMb8pBeqJBNGx81uuf7iljX9++yNBVRDQgD04L+SVXmXmh5ZP4/WOp4mWF0kmi06PW2tA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@types/babel__core": {
+ "version": "7.20.5",
+ "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz",
+ "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/parser": "^7.20.7",
+ "@babel/types": "^7.20.7",
+ "@types/babel__generator": "*",
+ "@types/babel__template": "*",
+ "@types/babel__traverse": "*"
+ }
+ },
+ "node_modules/@types/babel__generator": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz",
+ "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/types": "^7.0.0"
+ }
+ },
+ "node_modules/@types/babel__template": {
+ "version": "7.4.4",
+ "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz",
+ "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/parser": "^7.1.0",
+ "@babel/types": "^7.0.0"
+ }
+ },
+ "node_modules/@types/babel__traverse": {
+ "version": "7.28.0",
+ "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz",
+ "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/types": "^7.28.2"
+ }
+ },
+ "node_modules/@types/estree": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
+ "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@types/history": {
+ "version": "4.7.11",
+ "resolved": "https://registry.npmjs.org/@types/history/-/history-4.7.11.tgz",
+ "integrity": "sha512-qjDJRrmvBMiTx+jyLxvLfJU7UznFuokDv4f3WRuriHKERccVpFU+8XMQUAbDzoiJCsmexxRExQeMwwCdamSKDA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@types/json-schema": {
+ "version": "7.0.15",
+ "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
+ "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@types/node": {
+ "version": "24.10.1",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-24.10.1.tgz",
+ "integrity": "sha512-GNWcUTRBgIRJD5zj+Tq0fKOJ5XZajIiBroOF0yvj2bSU1WvNdYS/dn9UxwsujGW4JX06dnHyjV2y9rRaybH0iQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "undici-types": "~7.16.0"
+ }
+ },
+ "node_modules/@types/react": {
+ "version": "19.2.5",
+ "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.5.tgz",
+ "integrity": "sha512-keKxkZMqnDicuvFoJbzrhbtdLSPhj/rZThDlKWCDbgXmUg0rEUFtRssDXKYmtXluZlIqiC5VqkCgRwzuyLHKHw==",
+ "devOptional": true,
+ "license": "MIT",
+ "dependencies": {
+ "csstype": "^3.0.2"
+ }
+ },
+ "node_modules/@types/react-dom": {
+ "version": "19.2.3",
+ "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz",
+ "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==",
+ "dev": true,
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "^19.2.0"
+ }
+ },
+ "node_modules/@types/react-router": {
+ "version": "5.1.20",
+ "resolved": "https://registry.npmjs.org/@types/react-router/-/react-router-5.1.20.tgz",
+ "integrity": "sha512-jGjmu/ZqS7FjSH6owMcD5qpq19+1RS9DeVRqfl1FeBMxTDQAGwlMWOcs52NDoXaNKyG3d1cYQFMs9rCrb88o9Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/history": "^4.7.11",
+ "@types/react": "*"
+ }
+ },
+ "node_modules/@types/react-router-dom": {
+ "version": "5.3.3",
+ "resolved": "https://registry.npmjs.org/@types/react-router-dom/-/react-router-dom-5.3.3.tgz",
+ "integrity": "sha512-kpqnYK4wcdm5UaWI3fLcELopqLrHgLqNsdpHauzlQktfkHL3npOSwtj1Uz9oKBAzs7lFtVkV8j83voAz2D8fhw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/history": "^4.7.11",
+ "@types/react": "*",
+ "@types/react-router": "*"
+ }
+ },
+ "node_modules/@typescript-eslint/eslint-plugin": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.46.4.tgz",
+ "integrity": "sha512-R48VhmTJqplNyDxCyqqVkFSZIx1qX6PzwqgcXn1olLrzxcSBDlOsbtcnQuQhNtnNiJ4Xe5gREI1foajYaYU2Vg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@eslint-community/regexpp": "^4.10.0",
+ "@typescript-eslint/scope-manager": "8.46.4",
+ "@typescript-eslint/type-utils": "8.46.4",
+ "@typescript-eslint/utils": "8.46.4",
+ "@typescript-eslint/visitor-keys": "8.46.4",
+ "graphemer": "^1.4.0",
+ "ignore": "^7.0.0",
+ "natural-compare": "^1.4.0",
+ "ts-api-utils": "^2.1.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "@typescript-eslint/parser": "^8.46.4",
+ "eslint": "^8.57.0 || ^9.0.0",
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": {
+ "version": "7.0.5",
+ "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz",
+ "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 4"
+ }
+ },
+ "node_modules/@typescript-eslint/parser": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.46.4.tgz",
+ "integrity": "sha512-tK3GPFWbirvNgsNKto+UmB/cRtn6TZfyw0D6IKrW55n6Vbs7KJoZtI//kpTKzE/DUmmnAFD8/Ca46s7Obs92/w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/scope-manager": "8.46.4",
+ "@typescript-eslint/types": "8.46.4",
+ "@typescript-eslint/typescript-estree": "8.46.4",
+ "@typescript-eslint/visitor-keys": "8.46.4",
+ "debug": "^4.3.4"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^8.57.0 || ^9.0.0",
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/project-service": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.46.4.tgz",
+ "integrity": "sha512-nPiRSKuvtTN+no/2N1kt2tUh/HoFzeEgOm9fQ6XQk4/ApGqjx0zFIIaLJ6wooR1HIoozvj2j6vTi/1fgAz7UYQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/tsconfig-utils": "^8.46.4",
+ "@typescript-eslint/types": "^8.46.4",
+ "debug": "^4.3.4"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/scope-manager": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.46.4.tgz",
+ "integrity": "sha512-tMDbLGXb1wC+McN1M6QeDx7P7c0UWO5z9CXqp7J8E+xGcJuUuevWKxuG8j41FoweS3+L41SkyKKkia16jpX7CA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/types": "8.46.4",
+ "@typescript-eslint/visitor-keys": "8.46.4"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ }
+ },
+ "node_modules/@typescript-eslint/tsconfig-utils": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.46.4.tgz",
+ "integrity": "sha512-+/XqaZPIAk6Cjg7NWgSGe27X4zMGqrFqZ8atJsX3CWxH/jACqWnrWI68h7nHQld0y+k9eTTjb9r+KU4twLoo9A==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/type-utils": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.46.4.tgz",
+ "integrity": "sha512-V4QC8h3fdT5Wro6vANk6eojqfbv5bpwHuMsBcJUJkqs2z5XnYhJzyz9Y02eUmF9u3PgXEUiOt4w4KHR3P+z0PQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/types": "8.46.4",
+ "@typescript-eslint/typescript-estree": "8.46.4",
+ "@typescript-eslint/utils": "8.46.4",
+ "debug": "^4.3.4",
+ "ts-api-utils": "^2.1.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^8.57.0 || ^9.0.0",
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/types": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.46.4.tgz",
+ "integrity": "sha512-USjyxm3gQEePdUwJBFjjGNG18xY9A2grDVGuk7/9AkjIF1L+ZrVnwR5VAU5JXtUnBL/Nwt3H31KlRDaksnM7/w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ }
+ },
+ "node_modules/@typescript-eslint/typescript-estree": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.46.4.tgz",
+ "integrity": "sha512-7oV2qEOr1d4NWNmpXLR35LvCfOkTNymY9oyW+lUHkmCno7aOmIf/hMaydnJBUTBMRCOGZh8YjkFOc8dadEoNGA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/project-service": "8.46.4",
+ "@typescript-eslint/tsconfig-utils": "8.46.4",
+ "@typescript-eslint/types": "8.46.4",
+ "@typescript-eslint/visitor-keys": "8.46.4",
+ "debug": "^4.3.4",
+ "fast-glob": "^3.3.2",
+ "is-glob": "^4.0.3",
+ "minimatch": "^9.0.4",
+ "semver": "^7.6.0",
+ "ts-api-utils": "^2.1.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz",
+ "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "balanced-match": "^1.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": {
+ "version": "9.0.5",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz",
+ "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "brace-expansion": "^2.0.1"
+ },
+ "engines": {
+ "node": ">=16 || 14 >=14.17"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": {
+ "version": "7.7.3",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz",
+ "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==",
+ "dev": true,
+ "license": "ISC",
+ "bin": {
+ "semver": "bin/semver.js"
+ },
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/@typescript-eslint/utils": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.46.4.tgz",
+ "integrity": "sha512-AbSv11fklGXV6T28dp2Me04Uw90R2iJ30g2bgLz529Koehrmkbs1r7paFqr1vPCZi7hHwYxYtxfyQMRC8QaVSg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@eslint-community/eslint-utils": "^4.7.0",
+ "@typescript-eslint/scope-manager": "8.46.4",
+ "@typescript-eslint/types": "8.46.4",
+ "@typescript-eslint/typescript-estree": "8.46.4"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^8.57.0 || ^9.0.0",
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/visitor-keys": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.46.4.tgz",
+ "integrity": "sha512-/++5CYLQqsO9HFGLI7APrxBJYo+5OCMpViuhV8q5/Qa3o5mMrF//eQHks+PXcsAVaLdn817fMuS7zqoXNNZGaw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/types": "8.46.4",
+ "eslint-visitor-keys": "^4.2.1"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ }
+ },
+ "node_modules/@vitejs/plugin-react": {
+ "version": "5.1.1",
+ "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-5.1.1.tgz",
+ "integrity": "sha512-WQfkSw0QbQ5aJ2CHYw23ZGkqnRwqKHD/KYsMeTkZzPT4Jcf0DcBxBtwMJxnu6E7oxw5+JC6ZAiePgh28uJ1HBA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/core": "^7.28.5",
+ "@babel/plugin-transform-react-jsx-self": "^7.27.1",
+ "@babel/plugin-transform-react-jsx-source": "^7.27.1",
+ "@rolldown/pluginutils": "1.0.0-beta.47",
+ "@types/babel__core": "^7.20.5",
+ "react-refresh": "^0.18.0"
+ },
+ "engines": {
+ "node": "^20.19.0 || >=22.12.0"
+ },
+ "peerDependencies": {
+ "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0"
+ }
+ },
+ "node_modules/acorn": {
+ "version": "8.15.0",
+ "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz",
+ "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "acorn": "bin/acorn"
+ },
+ "engines": {
+ "node": ">=0.4.0"
+ }
+ },
+ "node_modules/acorn-jsx": {
+ "version": "5.3.2",
+ "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz",
+ "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==",
+ "dev": true,
+ "license": "MIT",
+ "peerDependencies": {
+ "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
+ }
+ },
+ "node_modules/ajv": {
+ "version": "6.12.6",
+ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+ "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "fast-deep-equal": "^3.1.1",
+ "fast-json-stable-stringify": "^2.0.0",
+ "json-schema-traverse": "^0.4.1",
+ "uri-js": "^4.2.2"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/epoberezkin"
+ }
+ },
+ "node_modules/ansi-styles": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
+ "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "color-convert": "^2.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/ansi-styles?sponsor=1"
+ }
+ },
+ "node_modules/argparse": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz",
+ "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==",
+ "dev": true,
+ "license": "Python-2.0"
+ },
+ "node_modules/asynckit": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
+ "license": "MIT"
+ },
+ "node_modules/axios": {
+ "version": "1.13.2",
+ "resolved": "https://registry.npmjs.org/axios/-/axios-1.13.2.tgz",
+ "integrity": "sha512-VPk9ebNqPcy5lRGuSlKx752IlDatOjT9paPlm8A7yOuW2Fbvp4X3JznJtT4f0GzGLLiWE9W8onz51SqLYwzGaA==",
+ "license": "MIT",
+ "dependencies": {
+ "follow-redirects": "^1.15.6",
+ "form-data": "^4.0.4",
+ "proxy-from-env": "^1.1.0"
+ }
+ },
+ "node_modules/balanced-match": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
+ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/baseline-browser-mapping": {
+ "version": "2.8.28",
+ "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.8.28.tgz",
+ "integrity": "sha512-gYjt7OIqdM0PcttNYP2aVrr2G0bMALkBaoehD4BuRGjAOtipg0b6wHg1yNL+s5zSnLZZrGHOw4IrND8CD+3oIQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "bin": {
+ "baseline-browser-mapping": "dist/cli.js"
+ }
+ },
+ "node_modules/brace-expansion": {
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "balanced-match": "^1.0.0",
+ "concat-map": "0.0.1"
+ }
+ },
+ "node_modules/braces": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz",
+ "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "fill-range": "^7.1.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/browserslist": {
+ "version": "4.28.0",
+ "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.0.tgz",
+ "integrity": "sha512-tbydkR/CxfMwelN0vwdP/pLkDwyAASZ+VfWm4EOwlB6SWhx1sYnWLqo8N5j0rAzPfzfRaxt0mM/4wPU/Su84RQ==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "baseline-browser-mapping": "^2.8.25",
+ "caniuse-lite": "^1.0.30001754",
+ "electron-to-chromium": "^1.5.249",
+ "node-releases": "^2.0.27",
+ "update-browserslist-db": "^1.1.4"
+ },
+ "bin": {
+ "browserslist": "cli.js"
+ },
+ "engines": {
+ "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
+ }
+ },
+ "node_modules/call-bind-apply-helpers": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
+ "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/callsites": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
+ "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/caniuse-lite": {
+ "version": "1.0.30001755",
+ "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001755.tgz",
+ "integrity": "sha512-44V+Jm6ctPj7R52Na4TLi3Zri4dWUljJd+RDm+j8LtNCc/ihLCT+X1TzoOAkRETEWqjuLnh9581Tl80FvK7jVA==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/caniuse-lite"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "CC-BY-4.0"
+ },
+ "node_modules/chalk": {
+ "version": "4.1.2",
+ "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
+ "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-styles": "^4.1.0",
+ "supports-color": "^7.1.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/chalk?sponsor=1"
+ }
+ },
+ "node_modules/color-convert": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
+ "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "color-name": "~1.1.4"
+ },
+ "engines": {
+ "node": ">=7.0.0"
+ }
+ },
+ "node_modules/color-name": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
+ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/combined-stream": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+ "license": "MIT",
+ "dependencies": {
+ "delayed-stream": "~1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/concat-map": {
+ "version": "0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+ "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/convert-source-map": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz",
+ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/cookie": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.0.2.tgz",
+ "integrity": "sha512-9Kr/j4O16ISv8zBBhJoi4bXOYNTkFLOqSL3UDB0njXxCXNezjeyVrJyGOWtgfs/q2km1gwBcfH8q1yEGoMYunA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/cross-spawn": {
+ "version": "7.0.6",
+ "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
+ "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "path-key": "^3.1.0",
+ "shebang-command": "^2.0.0",
+ "which": "^2.0.1"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/csstype": {
+ "version": "3.2.3",
+ "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz",
+ "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==",
+ "devOptional": true,
+ "license": "MIT"
+ },
+ "node_modules/debug": {
+ "version": "4.4.3",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
+ "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ms": "^2.1.3"
+ },
+ "engines": {
+ "node": ">=6.0"
+ },
+ "peerDependenciesMeta": {
+ "supports-color": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/deep-is": {
+ "version": "0.1.4",
+ "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
+ "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/delayed-stream": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.4.0"
+ }
+ },
+ "node_modules/dunder-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
+ "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
+ "license": "MIT",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "gopd": "^1.2.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/electron-to-chromium": {
+ "version": "1.5.254",
+ "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.254.tgz",
+ "integrity": "sha512-DcUsWpVhv9svsKRxnSCZ86SjD+sp32SGidNB37KpqXJncp1mfUgKbHvBomE89WJDbfVKw1mdv5+ikrvd43r+Bg==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/es-define-property": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
+ "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-errors": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-object-atoms": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
+ "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-set-tostringtag": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz",
+ "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "get-intrinsic": "^1.2.6",
+ "has-tostringtag": "^1.0.2",
+ "hasown": "^2.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/esbuild": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz",
+ "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==",
+ "dev": true,
+ "hasInstallScript": true,
+ "license": "MIT",
+ "bin": {
+ "esbuild": "bin/esbuild"
+ },
+ "engines": {
+ "node": ">=18"
+ },
+ "optionalDependencies": {
+ "@esbuild/aix-ppc64": "0.25.12",
+ "@esbuild/android-arm": "0.25.12",
+ "@esbuild/android-arm64": "0.25.12",
+ "@esbuild/android-x64": "0.25.12",
+ "@esbuild/darwin-arm64": "0.25.12",
+ "@esbuild/darwin-x64": "0.25.12",
+ "@esbuild/freebsd-arm64": "0.25.12",
+ "@esbuild/freebsd-x64": "0.25.12",
+ "@esbuild/linux-arm": "0.25.12",
+ "@esbuild/linux-arm64": "0.25.12",
+ "@esbuild/linux-ia32": "0.25.12",
+ "@esbuild/linux-loong64": "0.25.12",
+ "@esbuild/linux-mips64el": "0.25.12",
+ "@esbuild/linux-ppc64": "0.25.12",
+ "@esbuild/linux-riscv64": "0.25.12",
+ "@esbuild/linux-s390x": "0.25.12",
+ "@esbuild/linux-x64": "0.25.12",
+ "@esbuild/netbsd-arm64": "0.25.12",
+ "@esbuild/netbsd-x64": "0.25.12",
+ "@esbuild/openbsd-arm64": "0.25.12",
+ "@esbuild/openbsd-x64": "0.25.12",
+ "@esbuild/openharmony-arm64": "0.25.12",
+ "@esbuild/sunos-x64": "0.25.12",
+ "@esbuild/win32-arm64": "0.25.12",
+ "@esbuild/win32-ia32": "0.25.12",
+ "@esbuild/win32-x64": "0.25.12"
+ }
+ },
+ "node_modules/escalade": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
+ "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/escape-string-regexp": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
+ "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/eslint": {
+ "version": "9.39.1",
+ "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.39.1.tgz",
+ "integrity": "sha512-BhHmn2yNOFA9H9JmmIVKJmd288g9hrVRDkdoIgRCRuSySRUHH7r/DI6aAXW9T1WwUuY3DFgrcaqB+deURBLR5g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@eslint-community/eslint-utils": "^4.8.0",
+ "@eslint-community/regexpp": "^4.12.1",
+ "@eslint/config-array": "^0.21.1",
+ "@eslint/config-helpers": "^0.4.2",
+ "@eslint/core": "^0.17.0",
+ "@eslint/eslintrc": "^3.3.1",
+ "@eslint/js": "9.39.1",
+ "@eslint/plugin-kit": "^0.4.1",
+ "@humanfs/node": "^0.16.6",
+ "@humanwhocodes/module-importer": "^1.0.1",
+ "@humanwhocodes/retry": "^0.4.2",
+ "@types/estree": "^1.0.6",
+ "ajv": "^6.12.4",
+ "chalk": "^4.0.0",
+ "cross-spawn": "^7.0.6",
+ "debug": "^4.3.2",
+ "escape-string-regexp": "^4.0.0",
+ "eslint-scope": "^8.4.0",
+ "eslint-visitor-keys": "^4.2.1",
+ "espree": "^10.4.0",
+ "esquery": "^1.5.0",
+ "esutils": "^2.0.2",
+ "fast-deep-equal": "^3.1.3",
+ "file-entry-cache": "^8.0.0",
+ "find-up": "^5.0.0",
+ "glob-parent": "^6.0.2",
+ "ignore": "^5.2.0",
+ "imurmurhash": "^0.1.4",
+ "is-glob": "^4.0.0",
+ "json-stable-stringify-without-jsonify": "^1.0.1",
+ "lodash.merge": "^4.6.2",
+ "minimatch": "^3.1.2",
+ "natural-compare": "^1.4.0",
+ "optionator": "^0.9.3"
+ },
+ "bin": {
+ "eslint": "bin/eslint.js"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://eslint.org/donate"
+ },
+ "peerDependencies": {
+ "jiti": "*"
+ },
+ "peerDependenciesMeta": {
+ "jiti": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/eslint-plugin-react-hooks": {
+ "version": "7.0.1",
+ "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-7.0.1.tgz",
+ "integrity": "sha512-O0d0m04evaNzEPoSW+59Mezf8Qt0InfgGIBJnpC0h3NH/WjUAR7BIKUfysC6todmtiZ/A0oUVS8Gce0WhBrHsA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/core": "^7.24.4",
+ "@babel/parser": "^7.24.4",
+ "hermes-parser": "^0.25.1",
+ "zod": "^3.25.0 || ^4.0.0",
+ "zod-validation-error": "^3.5.0 || ^4.0.0"
+ },
+ "engines": {
+ "node": ">=18"
+ },
+ "peerDependencies": {
+ "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0"
+ }
+ },
+ "node_modules/eslint-plugin-react-refresh": {
+ "version": "0.4.24",
+ "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.24.tgz",
+ "integrity": "sha512-nLHIW7TEq3aLrEYWpVaJ1dRgFR+wLDPN8e8FpYAql/bMV2oBEfC37K0gLEGgv9fy66juNShSMV8OkTqzltcG/w==",
+ "dev": true,
+ "license": "MIT",
+ "peerDependencies": {
+ "eslint": ">=8.40"
+ }
+ },
+ "node_modules/eslint-scope": {
+ "version": "8.4.0",
+ "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz",
+ "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "esrecurse": "^4.3.0",
+ "estraverse": "^5.2.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/eslint-visitor-keys": {
+ "version": "4.2.1",
+ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz",
+ "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/espree": {
+ "version": "10.4.0",
+ "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz",
+ "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "acorn": "^8.15.0",
+ "acorn-jsx": "^5.3.2",
+ "eslint-visitor-keys": "^4.2.1"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/esquery": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz",
+ "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "dependencies": {
+ "estraverse": "^5.1.0"
+ },
+ "engines": {
+ "node": ">=0.10"
+ }
+ },
+ "node_modules/esrecurse": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz",
+ "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "estraverse": "^5.2.0"
+ },
+ "engines": {
+ "node": ">=4.0"
+ }
+ },
+ "node_modules/estraverse": {
+ "version": "5.3.0",
+ "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+ "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "engines": {
+ "node": ">=4.0"
+ }
+ },
+ "node_modules/esutils": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
+ "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/fast-deep-equal": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
+ "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/fast-glob": {
+ "version": "3.3.3",
+ "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz",
+ "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@nodelib/fs.stat": "^2.0.2",
+ "@nodelib/fs.walk": "^1.2.3",
+ "glob-parent": "^5.1.2",
+ "merge2": "^1.3.0",
+ "micromatch": "^4.0.8"
+ },
+ "engines": {
+ "node": ">=8.6.0"
+ }
+ },
+ "node_modules/fast-glob/node_modules/glob-parent": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
+ "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "is-glob": "^4.0.1"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/fast-json-stable-stringify": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
+ "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/fast-levenshtein": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
+ "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/fastq": {
+ "version": "1.19.1",
+ "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz",
+ "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "reusify": "^1.0.4"
+ }
+ },
+ "node_modules/file-entry-cache": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz",
+ "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "flat-cache": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=16.0.0"
+ }
+ },
+ "node_modules/fill-range": {
+ "version": "7.1.1",
+ "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz",
+ "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "to-regex-range": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/find-up": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+ "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "locate-path": "^6.0.0",
+ "path-exists": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/flat-cache": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz",
+ "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "flatted": "^3.2.9",
+ "keyv": "^4.5.4"
+ },
+ "engines": {
+ "node": ">=16"
+ }
+ },
+ "node_modules/flatted": {
+ "version": "3.3.3",
+ "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz",
+ "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/follow-redirects": {
+ "version": "1.15.11",
+ "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.11.tgz",
+ "integrity": "sha512-deG2P0JfjrTxl50XGCDyfI97ZGVCxIpfKYmfyrQ54n5FO/0gfIES8C/Psl6kWVDolizcaaxZJnTS0QSMxvnsBQ==",
+ "funding": [
+ {
+ "type": "individual",
+ "url": "https://github.com/sponsors/RubenVerborgh"
+ }
+ ],
+ "license": "MIT",
+ "engines": {
+ "node": ">=4.0"
+ },
+ "peerDependenciesMeta": {
+ "debug": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/form-data": {
+ "version": "4.0.5",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.5.tgz",
+ "integrity": "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w==",
+ "license": "MIT",
+ "dependencies": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.8",
+ "es-set-tostringtag": "^2.1.0",
+ "hasown": "^2.0.2",
+ "mime-types": "^2.1.12"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/fsevents": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
+ "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
+ "dev": true,
+ "hasInstallScript": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
+ }
+ },
+ "node_modules/function-bind": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
+ "license": "MIT",
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/gensync": {
+ "version": "1.0.0-beta.2",
+ "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz",
+ "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/get-intrinsic": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
+ "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
+ "license": "MIT",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.2",
+ "es-define-property": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "es-object-atoms": "^1.1.1",
+ "function-bind": "^1.1.2",
+ "get-proto": "^1.0.1",
+ "gopd": "^1.2.0",
+ "has-symbols": "^1.1.0",
+ "hasown": "^2.0.2",
+ "math-intrinsics": "^1.1.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/get-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
+ "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
+ "license": "MIT",
+ "dependencies": {
+ "dunder-proto": "^1.0.1",
+ "es-object-atoms": "^1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/glob-parent": {
+ "version": "6.0.2",
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
+ "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "is-glob": "^4.0.3"
+ },
+ "engines": {
+ "node": ">=10.13.0"
+ }
+ },
+ "node_modules/globals": {
+ "version": "16.5.0",
+ "resolved": "https://registry.npmjs.org/globals/-/globals-16.5.0.tgz",
+ "integrity": "sha512-c/c15i26VrJ4IRt5Z89DnIzCGDn9EcebibhAOjw5ibqEHsE1wLUgkPn9RDmNcUKyU87GeaL633nyJ+pplFR2ZQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/gopd": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
+ "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/graphemer": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz",
+ "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/has-flag": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
+ "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/has-symbols": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
+ "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-tostringtag": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz",
+ "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==",
+ "license": "MIT",
+ "dependencies": {
+ "has-symbols": "^1.0.3"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/hasown": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
+ "license": "MIT",
+ "dependencies": {
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/hermes-estree": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz",
+ "integrity": "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/hermes-parser": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.25.1.tgz",
+ "integrity": "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "hermes-estree": "0.25.1"
+ }
+ },
+ "node_modules/ignore": {
+ "version": "5.3.2",
+ "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz",
+ "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 4"
+ }
+ },
+ "node_modules/import-fresh": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz",
+ "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "parent-module": "^1.0.0",
+ "resolve-from": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=6"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/imurmurhash": {
+ "version": "0.1.4",
+ "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
+ "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.8.19"
+ }
+ },
+ "node_modules/is-extglob": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
+ "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/is-glob": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
+ "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "is-extglob": "^2.1.1"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/is-number": {
+ "version": "7.0.0",
+ "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
+ "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.12.0"
+ }
+ },
+ "node_modules/isexe": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
+ "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/js-tokens": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
+ "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/js-yaml": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz",
+ "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "argparse": "^2.0.1"
+ },
+ "bin": {
+ "js-yaml": "bin/js-yaml.js"
+ }
+ },
+ "node_modules/jsesc": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz",
+ "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "jsesc": "bin/jsesc"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/json-buffer": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz",
+ "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/json-schema-traverse": {
+ "version": "0.4.1",
+ "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+ "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/json-stable-stringify-without-jsonify": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
+ "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/json5": {
+ "version": "2.2.3",
+ "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz",
+ "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "json5": "lib/cli.js"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/keyv": {
+ "version": "4.5.4",
+ "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
+ "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "json-buffer": "3.0.1"
+ }
+ },
+ "node_modules/levn": {
+ "version": "0.4.1",
+ "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz",
+ "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "prelude-ls": "^1.2.1",
+ "type-check": "~0.4.0"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/locate-path": {
+ "version": "6.0.0",
+ "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+ "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "p-locate": "^5.0.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/lodash.merge": {
+ "version": "4.6.2",
+ "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
+ "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/lru-cache": {
+ "version": "5.1.1",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
+ "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "yallist": "^3.0.2"
+ }
+ },
+ "node_modules/lucide-react": {
+ "version": "0.554.0",
+ "resolved": "https://registry.npmjs.org/lucide-react/-/lucide-react-0.554.0.tgz",
+ "integrity": "sha512-St+z29uthEJVx0Is7ellNkgTEhaeSoA42I7JjOCBCrc5X6LYMGSv0P/2uS5HDLTExP5tpiqRD2PyUEOS6s9UXA==",
+ "license": "ISC",
+ "peerDependencies": {
+ "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/math-intrinsics": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
+ "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/merge2": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz",
+ "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/micromatch": {
+ "version": "4.0.8",
+ "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz",
+ "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "braces": "^3.0.3",
+ "picomatch": "^2.3.1"
+ },
+ "engines": {
+ "node": ">=8.6"
+ }
+ },
+ "node_modules/mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/mime-types": {
+ "version": "2.1.35",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+ "license": "MIT",
+ "dependencies": {
+ "mime-db": "1.52.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/minimatch": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+ "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "brace-expansion": "^1.1.7"
+ },
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/nanoid": {
+ "version": "3.3.11",
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
+ "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "bin": {
+ "nanoid": "bin/nanoid.cjs"
+ },
+ "engines": {
+ "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
+ }
+ },
+ "node_modules/natural-compare": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
+ "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/node-releases": {
+ "version": "2.0.27",
+ "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz",
+ "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/optionator": {
+ "version": "0.9.4",
+ "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz",
+ "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "deep-is": "^0.1.3",
+ "fast-levenshtein": "^2.0.6",
+ "levn": "^0.4.1",
+ "prelude-ls": "^1.2.1",
+ "type-check": "^0.4.0",
+ "word-wrap": "^1.2.5"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/p-limit": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+ "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "yocto-queue": "^0.1.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/p-locate": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+ "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "p-limit": "^3.0.2"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/parent-module": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz",
+ "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "callsites": "^3.0.0"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/path-exists": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+ "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/path-key": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
+ "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/picocolors": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
+ "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/picomatch": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz",
+ "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8.6"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/jonschlinkert"
+ }
+ },
+ "node_modules/postcss": {
+ "version": "8.5.6",
+ "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz",
+ "integrity": "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/postcss"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "nanoid": "^3.3.11",
+ "picocolors": "^1.1.1",
+ "source-map-js": "^1.2.1"
+ },
+ "engines": {
+ "node": "^10 || ^12 || >=14"
+ }
+ },
+ "node_modules/prelude-ls": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
+ "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/proxy-from-env": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+ "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
+ "license": "MIT"
+ },
+ "node_modules/punycode": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
+ "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/queue-microtask": {
+ "version": "1.2.3",
+ "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
+ "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT"
+ },
+ "node_modules/react": {
+ "version": "19.2.0",
+ "resolved": "https://registry.npmjs.org/react/-/react-19.2.0.tgz",
+ "integrity": "sha512-tmbWg6W31tQLeB5cdIBOicJDJRR2KzXsV7uSK9iNfLWQ5bIZfxuPEHp7M8wiHyHnn0DD1i7w3Zmin0FtkrwoCQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/react-dom": {
+ "version": "19.2.0",
+ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.0.tgz",
+ "integrity": "sha512-UlbRu4cAiGaIewkPyiRGJk0imDN2T3JjieT6spoL2UeSf5od4n5LB/mQ4ejmxhCFT1tYe8IvaFulzynWovsEFQ==",
+ "license": "MIT",
+ "dependencies": {
+ "scheduler": "^0.27.0"
+ },
+ "peerDependencies": {
+ "react": "^19.2.0"
+ }
+ },
+ "node_modules/react-hook-form": {
+ "version": "7.66.0",
+ "resolved": "https://registry.npmjs.org/react-hook-form/-/react-hook-form-7.66.0.tgz",
+ "integrity": "sha512-xXBqsWGKrY46ZqaHDo+ZUYiMUgi8suYu5kdrS20EG8KiL7VRQitEbNjm+UcrDYrNi1YLyfpmAeGjCZYXLT9YBw==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=18.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/react-hook-form"
+ },
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17 || ^18 || ^19"
+ }
+ },
+ "node_modules/react-refresh": {
+ "version": "0.18.0",
+ "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.18.0.tgz",
+ "integrity": "sha512-QgT5//D3jfjJb6Gsjxv0Slpj23ip+HtOpnNgnb2S5zU3CB26G/IDPGoy4RJB42wzFE46DRsstbW6tKHoKbhAxw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/react-router": {
+ "version": "7.9.6",
+ "resolved": "https://registry.npmjs.org/react-router/-/react-router-7.9.6.tgz",
+ "integrity": "sha512-Y1tUp8clYRXpfPITyuifmSoE2vncSME18uVLgaqyxh9H35JWpIfzHo+9y3Fzh5odk/jxPW29IgLgzcdwxGqyNA==",
+ "license": "MIT",
+ "dependencies": {
+ "cookie": "^1.0.1",
+ "set-cookie-parser": "^2.6.0"
+ },
+ "engines": {
+ "node": ">=20.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=18",
+ "react-dom": ">=18"
+ },
+ "peerDependenciesMeta": {
+ "react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/react-router-dom": {
+ "version": "7.9.6",
+ "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-7.9.6.tgz",
+ "integrity": "sha512-2MkC2XSXq6HjGcihnx1s0DBWQETI4mlis4Ux7YTLvP67xnGxCvq+BcCQSO81qQHVUTM1V53tl4iVVaY5sReCOA==",
+ "license": "MIT",
+ "dependencies": {
+ "react-router": "7.9.6"
+ },
+ "engines": {
+ "node": ">=20.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=18",
+ "react-dom": ">=18"
+ }
+ },
+ "node_modules/resolve-from": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz",
+ "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/reusify": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz",
+ "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "iojs": ">=1.0.0",
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/rollup": {
+ "version": "4.53.2",
+ "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.53.2.tgz",
+ "integrity": "sha512-MHngMYwGJVi6Fmnk6ISmnk7JAHRNF0UkuucA0CUW3N3a4KnONPEZz+vUanQP/ZC/iY1Qkf3bwPWzyY84wEks1g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/estree": "1.0.8"
+ },
+ "bin": {
+ "rollup": "dist/bin/rollup"
+ },
+ "engines": {
+ "node": ">=18.0.0",
+ "npm": ">=8.0.0"
+ },
+ "optionalDependencies": {
+ "@rollup/rollup-android-arm-eabi": "4.53.2",
+ "@rollup/rollup-android-arm64": "4.53.2",
+ "@rollup/rollup-darwin-arm64": "4.53.2",
+ "@rollup/rollup-darwin-x64": "4.53.2",
+ "@rollup/rollup-freebsd-arm64": "4.53.2",
+ "@rollup/rollup-freebsd-x64": "4.53.2",
+ "@rollup/rollup-linux-arm-gnueabihf": "4.53.2",
+ "@rollup/rollup-linux-arm-musleabihf": "4.53.2",
+ "@rollup/rollup-linux-arm64-gnu": "4.53.2",
+ "@rollup/rollup-linux-arm64-musl": "4.53.2",
+ "@rollup/rollup-linux-loong64-gnu": "4.53.2",
+ "@rollup/rollup-linux-ppc64-gnu": "4.53.2",
+ "@rollup/rollup-linux-riscv64-gnu": "4.53.2",
+ "@rollup/rollup-linux-riscv64-musl": "4.53.2",
+ "@rollup/rollup-linux-s390x-gnu": "4.53.2",
+ "@rollup/rollup-linux-x64-gnu": "4.53.2",
+ "@rollup/rollup-linux-x64-musl": "4.53.2",
+ "@rollup/rollup-openharmony-arm64": "4.53.2",
+ "@rollup/rollup-win32-arm64-msvc": "4.53.2",
+ "@rollup/rollup-win32-ia32-msvc": "4.53.2",
+ "@rollup/rollup-win32-x64-gnu": "4.53.2",
+ "@rollup/rollup-win32-x64-msvc": "4.53.2",
+ "fsevents": "~2.3.2"
+ }
+ },
+ "node_modules/run-parallel": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz",
+ "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "queue-microtask": "^1.2.2"
+ }
+ },
+ "node_modules/scheduler": {
+ "version": "0.27.0",
+ "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz",
+ "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==",
+ "license": "MIT"
+ },
+ "node_modules/semver": {
+ "version": "6.3.1",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
+ "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
+ "dev": true,
+ "license": "ISC",
+ "bin": {
+ "semver": "bin/semver.js"
+ }
+ },
+ "node_modules/set-cookie-parser": {
+ "version": "2.7.2",
+ "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.2.tgz",
+ "integrity": "sha512-oeM1lpU/UvhTxw+g3cIfxXHyJRc/uidd3yK1P242gzHds0udQBYzs3y8j4gCCW+ZJ7ad0yctld8RYO+bdurlvw==",
+ "license": "MIT"
+ },
+ "node_modules/shebang-command": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
+ "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "shebang-regex": "^3.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/shebang-regex": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
+ "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/source-map-js": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
+ "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/strip-json-comments": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
+ "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/supports-color": {
+ "version": "7.2.0",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
+ "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "has-flag": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/tinyglobby": {
+ "version": "0.2.15",
+ "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.15.tgz",
+ "integrity": "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "fdir": "^6.5.0",
+ "picomatch": "^4.0.3"
+ },
+ "engines": {
+ "node": ">=12.0.0"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/SuperchupuDev"
+ }
+ },
+ "node_modules/tinyglobby/node_modules/fdir": {
+ "version": "6.5.0",
+ "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
+ "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=12.0.0"
+ },
+ "peerDependencies": {
+ "picomatch": "^3 || ^4"
+ },
+ "peerDependenciesMeta": {
+ "picomatch": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/tinyglobby/node_modules/picomatch": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
+ "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/jonschlinkert"
+ }
+ },
+ "node_modules/to-regex-range": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
+ "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "is-number": "^7.0.0"
+ },
+ "engines": {
+ "node": ">=8.0"
+ }
+ },
+ "node_modules/ts-api-utils": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.1.0.tgz",
+ "integrity": "sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=18.12"
+ },
+ "peerDependencies": {
+ "typescript": ">=4.8.4"
+ }
+ },
+ "node_modules/type-check": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz",
+ "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "prelude-ls": "^1.2.1"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/typescript": {
+ "version": "5.9.3",
+ "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
+ "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "bin": {
+ "tsc": "bin/tsc",
+ "tsserver": "bin/tsserver"
+ },
+ "engines": {
+ "node": ">=14.17"
+ }
+ },
+ "node_modules/typescript-eslint": {
+ "version": "8.46.4",
+ "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.46.4.tgz",
+ "integrity": "sha512-KALyxkpYV5Ix7UhvjTwJXZv76VWsHG+NjNlt/z+a17SOQSiOcBdUXdbJdyXi7RPxrBFECtFOiPwUJQusJuCqrg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/eslint-plugin": "8.46.4",
+ "@typescript-eslint/parser": "8.46.4",
+ "@typescript-eslint/typescript-estree": "8.46.4",
+ "@typescript-eslint/utils": "8.46.4"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^8.57.0 || ^9.0.0",
+ "typescript": ">=4.8.4 <6.0.0"
+ }
+ },
+ "node_modules/undici-types": {
+ "version": "7.16.0",
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz",
+ "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/update-browserslist-db": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.4.tgz",
+ "integrity": "sha512-q0SPT4xyU84saUX+tomz1WLkxUbuaJnR1xWt17M7fJtEJigJeWUNGUqrauFXsHnqev9y9JTRGwk13tFBuKby4A==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "escalade": "^3.2.0",
+ "picocolors": "^1.1.1"
+ },
+ "bin": {
+ "update-browserslist-db": "cli.js"
+ },
+ "peerDependencies": {
+ "browserslist": ">= 4.21.0"
+ }
+ },
+ "node_modules/uri-js": {
+ "version": "4.4.1",
+ "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
+ "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "punycode": "^2.1.0"
+ }
+ },
+ "node_modules/vite": {
+ "version": "7.2.2",
+ "resolved": "https://registry.npmjs.org/vite/-/vite-7.2.2.tgz",
+ "integrity": "sha512-BxAKBWmIbrDgrokdGZH1IgkIk/5mMHDreLDmCJ0qpyJaAteP8NvMhkwr/ZCQNqNH97bw/dANTE9PDzqwJghfMQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "esbuild": "^0.25.0",
+ "fdir": "^6.5.0",
+ "picomatch": "^4.0.3",
+ "postcss": "^8.5.6",
+ "rollup": "^4.43.0",
+ "tinyglobby": "^0.2.15"
+ },
+ "bin": {
+ "vite": "bin/vite.js"
+ },
+ "engines": {
+ "node": "^20.19.0 || >=22.12.0"
+ },
+ "funding": {
+ "url": "https://github.com/vitejs/vite?sponsor=1"
+ },
+ "optionalDependencies": {
+ "fsevents": "~2.3.3"
+ },
+ "peerDependencies": {
+ "@types/node": "^20.19.0 || >=22.12.0",
+ "jiti": ">=1.21.0",
+ "less": "^4.0.0",
+ "lightningcss": "^1.21.0",
+ "sass": "^1.70.0",
+ "sass-embedded": "^1.70.0",
+ "stylus": ">=0.54.8",
+ "sugarss": "^5.0.0",
+ "terser": "^5.16.0",
+ "tsx": "^4.8.1",
+ "yaml": "^2.4.2"
+ },
+ "peerDependenciesMeta": {
+ "@types/node": {
+ "optional": true
+ },
+ "jiti": {
+ "optional": true
+ },
+ "less": {
+ "optional": true
+ },
+ "lightningcss": {
+ "optional": true
+ },
+ "sass": {
+ "optional": true
+ },
+ "sass-embedded": {
+ "optional": true
+ },
+ "stylus": {
+ "optional": true
+ },
+ "sugarss": {
+ "optional": true
+ },
+ "terser": {
+ "optional": true
+ },
+ "tsx": {
+ "optional": true
+ },
+ "yaml": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/vite/node_modules/fdir": {
+ "version": "6.5.0",
+ "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
+ "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=12.0.0"
+ },
+ "peerDependencies": {
+ "picomatch": "^3 || ^4"
+ },
+ "peerDependenciesMeta": {
+ "picomatch": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/vite/node_modules/picomatch": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
+ "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/jonschlinkert"
+ }
+ },
+ "node_modules/which": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
+ "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "isexe": "^2.0.0"
+ },
+ "bin": {
+ "node-which": "bin/node-which"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/word-wrap": {
+ "version": "1.2.5",
+ "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz",
+ "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/yallist": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+ "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/yocto-queue": {
+ "version": "0.1.0",
+ "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
+ "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/zod": {
+ "version": "4.1.12",
+ "resolved": "https://registry.npmjs.org/zod/-/zod-4.1.12.tgz",
+ "integrity": "sha512-JInaHOamG8pt5+Ey8kGmdcAcg3OL9reK8ltczgHTAwNhMys/6ThXHityHxVV2p3fkw/c+MAvBHFVYHFZDmjMCQ==",
+ "dev": true,
+ "license": "MIT",
+ "funding": {
+ "url": "https://github.com/sponsors/colinhacks"
+ }
+ },
+ "node_modules/zod-validation-error": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/zod-validation-error/-/zod-validation-error-4.0.2.tgz",
+ "integrity": "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=18.0.0"
+ },
+ "peerDependencies": {
+ "zod": "^3.25.0 || ^4.0.0"
+ }
+ },
+ "node_modules/zustand": {
+ "version": "5.0.8",
+ "resolved": "https://registry.npmjs.org/zustand/-/zustand-5.0.8.tgz",
+ "integrity": "sha512-gyPKpIaxY9XcO2vSMrLbiER7QMAMGOQZVRdJ6Zi782jkbzZygq5GI9nG8g+sMgitRtndwaBSl7uiqC49o1SSiw==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=12.20.0"
+ },
+ "peerDependencies": {
+ "@types/react": ">=18.0.0",
+ "immer": ">=9.0.6",
+ "react": ">=18.0.0",
+ "use-sync-external-store": ">=1.2.0"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "immer": {
+ "optional": true
+ },
+ "react": {
+ "optional": true
+ },
+ "use-sync-external-store": {
+ "optional": true
+ }
+ }
+ }
+ }
+}
diff --git a/site-builder/package.json b/site-builder/package.json
new file mode 100644
index 00000000..19930e0b
--- /dev/null
+++ b/site-builder/package.json
@@ -0,0 +1,36 @@
+{
+ "name": "site-builder",
+ "private": true,
+ "version": "0.0.0",
+ "type": "module",
+ "scripts": {
+ "dev": "vite",
+ "build": "tsc -b && vite build",
+ "lint": "eslint .",
+ "preview": "vite preview"
+ },
+ "dependencies": {
+ "axios": "^1.13.2",
+ "lucide-react": "^0.554.0",
+ "react": "^19.2.0",
+ "react-dom": "^19.2.0",
+ "react-hook-form": "^7.66.0",
+ "react-router-dom": "^7.9.6",
+ "zustand": "^5.0.8"
+ },
+ "devDependencies": {
+ "@eslint/js": "^9.39.1",
+ "@types/node": "^24.10.0",
+ "@types/react": "^19.2.2",
+ "@types/react-dom": "^19.2.2",
+ "@types/react-router-dom": "^5.3.3",
+ "@vitejs/plugin-react": "^5.1.0",
+ "eslint": "^9.39.1",
+ "eslint-plugin-react-hooks": "^7.0.1",
+ "eslint-plugin-react-refresh": "^0.4.24",
+ "globals": "^16.5.0",
+ "typescript": "~5.9.3",
+ "typescript-eslint": "^8.46.3",
+ "vite": "^7.2.2"
+ }
+}
diff --git a/site-builder/public/vite.svg b/site-builder/public/vite.svg
new file mode 100644
index 00000000..e7b8dfb1
--- /dev/null
+++ b/site-builder/public/vite.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/site-builder/src/App.css b/site-builder/src/App.css
new file mode 100644
index 00000000..c9137d76
--- /dev/null
+++ b/site-builder/src/App.css
@@ -0,0 +1,330 @@
+.app-shell {
+ display: grid;
+ grid-template-columns: 280px 1fr;
+ min-height: 100vh;
+ background: #f5f7fb;
+ color: #0f172a;
+}
+
+.app-sidebar {
+ border-right: 1px solid rgba(15, 23, 42, 0.08);
+ padding: 2rem 1.5rem;
+ display: flex;
+ flex-direction: column;
+ gap: 2rem;
+ background: #ffffff;
+}
+
+.app-sidebar .brand span {
+ font-size: 1.25rem;
+ font-weight: 700;
+ display: block;
+}
+
+.app-sidebar .brand small {
+ color: #64748b;
+}
+
+.app-sidebar nav {
+ display: flex;
+ flex-direction: column;
+ gap: 0.75rem;
+}
+
+.app-sidebar a {
+ display: flex;
+ align-items: center;
+ gap: 0.75rem;
+ padding: 0.6rem 0.75rem;
+ border-radius: 10px;
+ text-decoration: none;
+ color: inherit;
+ font-weight: 500;
+}
+
+.app-sidebar a.active {
+ background: #eef2ff;
+ color: #4338ca;
+}
+
+.app-main {
+ padding: 2rem 3rem;
+ overflow-y: auto;
+}
+
+.wizard-page {
+ display: flex;
+ flex-direction: column;
+ gap: 1.5rem;
+}
+
+.wizard-progress {
+ display: flex;
+ gap: 1rem;
+ flex-wrap: wrap;
+}
+
+.wizard-progress__dot {
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ gap: 0.35rem;
+ border: none;
+ background: transparent;
+ color: #94a3b8;
+ font-weight: 500;
+ cursor: pointer;
+}
+
+.wizard-progress__dot span {
+ width: 34px;
+ height: 34px;
+ border-radius: 50%;
+ border: 2px solid currentColor;
+ display: grid;
+ place-items: center;
+}
+
+.wizard-progress__dot.is-active {
+ color: #4338ca;
+}
+
+.wizard-step {
+ margin-top: 1rem;
+}
+
+.wizard-actions {
+ display: flex;
+ justify-content: space-between;
+ gap: 1rem;
+}
+
+.wizard-actions button {
+ padding: 0.75rem 1.5rem;
+ border-radius: 12px;
+ border: 1px solid rgba(67, 56, 202, 0.3);
+ background: #fff;
+ cursor: pointer;
+}
+
+.wizard-actions button.primary {
+ background: #4338ca;
+ color: #fff;
+ border-color: transparent;
+ display: inline-flex;
+ align-items: center;
+ gap: 0.4rem;
+}
+
+.wizard-actions button.ghost,
+.ghost {
+ background: transparent;
+ border-color: rgba(67, 56, 202, 0.35);
+ color: #4338ca;
+}
+
+.wizard-actions button:disabled {
+ opacity: 0.5;
+ cursor: not-allowed;
+}
+
+.sb-error {
+ color: #dc2626;
+ margin: 0.5rem 0 0;
+}
+
+.sb-grid {
+ display: grid;
+ gap: 1rem;
+ grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
+}
+
+.sb-field {
+ display: flex;
+ flex-direction: column;
+ gap: 0.4rem;
+ font-size: 0.95rem;
+ color: #0f172a;
+}
+
+.sb-field input,
+.sb-field select,
+.sb-field textarea {
+ border: 1px solid rgba(15, 23, 42, 0.15);
+ border-radius: 10px;
+ padding: 0.65rem 0.85rem;
+ font-size: 0.95rem;
+ font-family: inherit;
+ background: #f8fafc;
+}
+
+.sb-pill-list {
+ display: flex;
+ flex-wrap: wrap;
+ gap: 0.5rem;
+}
+
+.sb-pill {
+ padding: 0.35rem 0.75rem;
+ border-radius: 999px;
+ background: rgba(67, 56, 202, 0.1);
+ color: #4338ca;
+ display: inline-flex;
+ align-items: center;
+ gap: 0.35rem;
+}
+
+.sb-pill button {
+ border: none;
+ background: transparent;
+ color: inherit;
+ cursor: pointer;
+}
+
+.sb-objective-input {
+ display: flex;
+ gap: 0.5rem;
+}
+
+.sb-objective-input input {
+ flex: 1;
+}
+
+.sb-objective-input button {
+ border: none;
+ background: #0f172a;
+ color: #fff;
+ border-radius: 10px;
+ padding: 0.65rem 1rem;
+ cursor: pointer;
+}
+
+.sb-blueprint-meta {
+ display: flex;
+ justify-content: space-between;
+ align-items: center;
+}
+
+.status-dot {
+ display: inline-flex;
+ align-items: center;
+ gap: 0.35rem;
+ text-transform: capitalize;
+}
+
+.status-dot::before {
+ content: '';
+ width: 10px;
+ height: 10px;
+ border-radius: 50%;
+ background: currentColor;
+}
+
+.status-ready {
+ color: #10b981;
+}
+
+.status-generating {
+ color: #f97316;
+}
+
+.status-draft {
+ color: #94a3b8;
+}
+
+.preview-canvas {
+ background: #fff;
+ border-radius: 18px;
+ padding: 1.5rem;
+ box-shadow: 0 8px 26px rgba(15, 23, 42, 0.08);
+}
+
+.preview-nav {
+ display: flex;
+ gap: 0.5rem;
+ flex-wrap: wrap;
+ margin-bottom: 1rem;
+}
+
+.preview-nav button {
+ border: 1px solid rgba(15, 23, 42, 0.1);
+ background: #f8fafc;
+ border-radius: 999px;
+ padding: 0.35rem 0.85rem;
+ cursor: pointer;
+}
+
+.preview-nav button.is-active {
+ background: #4338ca;
+ color: white;
+ border-color: transparent;
+}
+
+.preview-hero {
+ margin-bottom: 1.25rem;
+}
+
+.preview-hero .preview-label {
+ text-transform: uppercase;
+ font-size: 0.7rem;
+ letter-spacing: 0.08em;
+ color: #94a3b8;
+}
+
+.preview-blocks {
+ display: grid;
+ gap: 1rem;
+}
+
+.preview-block {
+ border: 1px dashed rgba(67, 56, 202, 0.2);
+ border-radius: 14px;
+ padding: 1rem;
+ background: rgba(67, 56, 202, 0.04);
+}
+
+.sb-blueprint-list {
+ list-style: none;
+ padding: 0;
+ margin: 0;
+ display: flex;
+ flex-direction: column;
+ gap: 1rem;
+}
+
+.sb-blueprint-list li {
+ display: flex;
+ justify-content: space-between;
+ align-items: center;
+ padding-bottom: 0.75rem;
+ border-bottom: 1px solid rgba(15, 23, 42, 0.08);
+}
+
+.sb-blueprint-list strong {
+ display: block;
+}
+
+.sb-blueprint-list span {
+ color: #475569;
+ font-size: 0.9rem;
+}
+
+.sb-loading {
+ display: flex;
+ align-items: center;
+ gap: 0.5rem;
+ color: #475569;
+}
+
+.spin {
+ animation: spin 1s linear infinite;
+}
+
+@keyframes spin {
+ from {
+ transform: rotate(0deg);
+ }
+ to {
+ transform: rotate(360deg);
+ }
+}
diff --git a/site-builder/src/App.tsx b/site-builder/src/App.tsx
new file mode 100644
index 00000000..cf68126b
--- /dev/null
+++ b/site-builder/src/App.tsx
@@ -0,0 +1,43 @@
+import { NavLink, Route, Routes } from 'react-router-dom';
+import { Wand2, LayoutTemplate, PanelsTopLeft } from 'lucide-react';
+import { WizardPage } from './pages/wizard/WizardPage';
+import { PreviewCanvas } from './pages/preview/PreviewCanvas';
+import { SiteDashboard } from './pages/dashboard/SiteDashboard';
+import './App.css';
+
+function App() {
+ return (
+
+
+
+
+
+ } />
+ } />
+ } />
+
+
+
+ );
+}
+
+export default App;
diff --git a/site-builder/src/api/builder.api.ts b/site-builder/src/api/builder.api.ts
new file mode 100644
index 00000000..30e7ab9b
--- /dev/null
+++ b/site-builder/src/api/builder.api.ts
@@ -0,0 +1,61 @@
+import axios from 'axios';
+import type {
+ BuilderFormData,
+ PageBlueprint,
+ SiteBlueprint,
+ SiteStructure,
+} from '../types/siteBuilder';
+
+const API_ROOT = import.meta.env.VITE_API_URL ?? 'http://localhost:8010/api';
+const BASE_PATH = `${API_ROOT}/v1/site-builder`;
+
+const client = axios.create({
+ baseURL: BASE_PATH,
+ withCredentials: true,
+});
+
+export interface CreateBlueprintPayload {
+ name: string;
+ description?: string;
+ site_id: number;
+ sector_id: number;
+ hosting_type: BuilderFormData['hostingType'];
+ config_json: Record;
+}
+
+export interface GenerateStructurePayload {
+ business_brief: string;
+ objectives: string[];
+ style: BuilderFormData['style'];
+ metadata?: Record;
+}
+
+export const builderApi = {
+ async listBlueprints(): Promise {
+ const res = await client.get('/blueprints/');
+ if (Array.isArray(res.data?.results)) {
+ return res.data.results as SiteBlueprint[];
+ }
+ return Array.isArray(res.data) ? res.data : [];
+ },
+
+ async createBlueprint(payload: CreateBlueprintPayload): Promise {
+ const res = await client.post('/blueprints/', payload);
+ return res.data;
+ },
+
+ async generateStructure(
+ blueprintId: number,
+ payload: GenerateStructurePayload,
+ ): Promise<{ task_id?: string; success?: boolean; structure?: SiteStructure }> {
+ const res = await client.post(`/blueprints/${blueprintId}/generate_structure/`, payload);
+ return res.data;
+ },
+
+ async listPages(blueprintId: number): Promise {
+ const res = await client.get(`/pages/?site_blueprint=${blueprintId}`);
+ return Array.isArray(res.data?.results) ? res.data.results : res.data;
+ },
+};
+
+
diff --git a/site-builder/src/assets/react.svg b/site-builder/src/assets/react.svg
new file mode 100644
index 00000000..6c87de9b
--- /dev/null
+++ b/site-builder/src/assets/react.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/site-builder/src/components/common/Card.css b/site-builder/src/components/common/Card.css
new file mode 100644
index 00000000..1ba25342
--- /dev/null
+++ b/site-builder/src/components/common/Card.css
@@ -0,0 +1,45 @@
+.sb-card {
+ background: #ffffff;
+ border-radius: 16px;
+ border: 1px solid rgba(15, 23, 42, 0.08);
+ padding: 1.5rem;
+ box-shadow: 0 8px 24px rgba(15, 23, 42, 0.05);
+ display: flex;
+ flex-direction: column;
+ gap: 1rem;
+}
+
+.sb-card__header {
+ display: flex;
+ flex-direction: column;
+ gap: 0.35rem;
+}
+
+.sb-card__title {
+ font-size: 1.1rem;
+ font-weight: 600;
+ color: #0f172a;
+ margin: 0;
+}
+
+.sb-card__description {
+ color: #475569;
+ margin: 0;
+ font-size: 0.95rem;
+}
+
+.sb-card__body {
+ display: flex;
+ flex-direction: column;
+ gap: 1rem;
+}
+
+.sb-card__footer {
+ border-top: 1px solid rgba(15, 23, 42, 0.06);
+ padding-top: 1rem;
+ display: flex;
+ justify-content: flex-end;
+ gap: 0.75rem;
+}
+
+
diff --git a/site-builder/src/components/common/Card.tsx b/site-builder/src/components/common/Card.tsx
new file mode 100644
index 00000000..841c48cf
--- /dev/null
+++ b/site-builder/src/components/common/Card.tsx
@@ -0,0 +1,25 @@
+import type { PropsWithChildren, ReactNode } from 'react';
+import './Card.css';
+
+interface CardProps extends PropsWithChildren {
+ title?: ReactNode;
+ description?: ReactNode;
+ footer?: ReactNode;
+}
+
+export function Card({ title, description, footer, children }: CardProps) {
+ return (
+
+ {(title || description) && (
+
+ {title && {title}
}
+ {description && {description}
}
+
+ )}
+ {children}
+ {footer && }
+
+ );
+}
+
+
diff --git a/site-builder/src/components/shared/blocks/HeroBlock.tsx b/site-builder/src/components/shared/blocks/HeroBlock.tsx
new file mode 100644
index 00000000..07424fbd
--- /dev/null
+++ b/site-builder/src/components/shared/blocks/HeroBlock.tsx
@@ -0,0 +1,28 @@
+import './HeroBlock.css';
+
+interface HeroBlockProps {
+ heading: string;
+ subheading?: string;
+ ctaLabel?: string;
+ secondaryCta?: string;
+ badge?: string;
+}
+
+export function HeroBlock({ heading, subheading, ctaLabel, secondaryCta, badge }: HeroBlockProps) {
+ return (
+
+ {badge &&
{badge}}
+
{heading}
+ {subheading &&
{subheading}
}
+
+ {ctaLabel && }
+ {secondaryCta && (
+
+ )}
+
+
+ );
+}
+
diff --git a/site-builder/src/components/shared/layouts/PageCanvas.css b/site-builder/src/components/shared/layouts/PageCanvas.css
new file mode 100644
index 00000000..b548b1b8
--- /dev/null
+++ b/site-builder/src/components/shared/layouts/PageCanvas.css
@@ -0,0 +1,13 @@
+.sb-page-canvas {
+ border-radius: 24px;
+ padding: 2.5rem;
+ box-shadow: 0 30px 80px rgba(15, 23, 42, 0.12);
+ border: 1px solid rgba(15, 23, 42, 0.08);
+}
+
+.sb-page-canvas__body {
+ display: flex;
+ flex-direction: column;
+ gap: 2.25rem;
+}
+
diff --git a/site-builder/src/components/shared/layouts/PageCanvas.tsx b/site-builder/src/components/shared/layouts/PageCanvas.tsx
new file mode 100644
index 00000000..907dd0b7
--- /dev/null
+++ b/site-builder/src/components/shared/layouts/PageCanvas.tsx
@@ -0,0 +1,12 @@
+import type { PropsWithChildren } from 'react';
+import { palette } from '../theme';
+import './PageCanvas.css';
+
+export function PageCanvas({ children }: PropsWithChildren) {
+ return (
+
+ {children}
+
+ );
+}
+
diff --git a/site-builder/src/components/shared/layouts/Section.css b/site-builder/src/components/shared/layouts/Section.css
new file mode 100644
index 00000000..20daac20
--- /dev/null
+++ b/site-builder/src/components/shared/layouts/Section.css
@@ -0,0 +1,39 @@
+.sb-section {
+ border-radius: 20px;
+ padding: 1.75rem;
+ border: 1px solid rgba(15, 23, 42, 0.07);
+ background: #ffffff;
+ display: flex;
+ flex-direction: column;
+ gap: 1.25rem;
+}
+
+.sb-section--soft {
+ background: #f4f6ff;
+}
+
+.sb-section__header h3 {
+ margin: 0.15rem 0;
+ font-size: 1.8rem;
+ font-weight: 700;
+}
+
+.sb-section__subtitle {
+ margin: 0;
+ color: #64748b;
+}
+
+.sb-section__overline {
+ font-size: 0.75rem;
+ letter-spacing: 0.08em;
+ text-transform: uppercase;
+ color: #6366f1;
+ margin: 0;
+}
+
+.sb-section__content {
+ display: flex;
+ flex-direction: column;
+ gap: 1rem;
+}
+
diff --git a/site-builder/src/components/shared/layouts/Section.tsx b/site-builder/src/components/shared/layouts/Section.tsx
new file mode 100644
index 00000000..d4cf74c2
--- /dev/null
+++ b/site-builder/src/components/shared/layouts/Section.tsx
@@ -0,0 +1,25 @@
+import type { PropsWithChildren, ReactNode } from 'react';
+import './Section.css';
+
+interface SectionProps extends PropsWithChildren {
+ overline?: string;
+ title?: ReactNode;
+ subtitle?: ReactNode;
+ background?: 'surface' | 'soft';
+}
+
+export function Section({ overline, title, subtitle, background = 'surface', children }: SectionProps) {
+ return (
+
+ {(overline || title || subtitle) && (
+
+ {overline && {overline}
}
+ {title && {title}
}
+ {subtitle && {subtitle}
}
+
+ )}
+ {children}
+
+ );
+}
+
diff --git a/site-builder/src/components/shared/theme.ts b/site-builder/src/components/shared/theme.ts
new file mode 100644
index 00000000..51ce000f
--- /dev/null
+++ b/site-builder/src/components/shared/theme.ts
@@ -0,0 +1,29 @@
+export const palette = {
+ background: '#f8fbff',
+ surface: '#ffffff',
+ accent: '#6366f1',
+ accentSoft: '#eef2ff',
+ text: '#0f172a',
+ textMuted: '#64748b',
+ border: 'rgba(15, 23, 42, 0.08)',
+};
+
+export const typography = {
+ title: {
+ fontSize: '2.5rem',
+ fontWeight: 700,
+ lineHeight: 1.1,
+ },
+ subtitle: {
+ fontSize: '1.15rem',
+ color: palette.textMuted,
+ lineHeight: 1.5,
+ },
+ label: {
+ fontSize: '0.75rem',
+ textTransform: 'uppercase',
+ letterSpacing: '0.08em',
+ color: palette.accent,
+ },
+};
+
diff --git a/site-builder/src/index.css b/site-builder/src/index.css
new file mode 100644
index 00000000..e4368115
--- /dev/null
+++ b/site-builder/src/index.css
@@ -0,0 +1,28 @@
+:root {
+ font-family: 'Inter', 'Inter var', system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
+ color: #0f172a;
+ background-color: #f5f7fb;
+ font-synthesis: none;
+ text-rendering: optimizeLegibility;
+ -webkit-font-smoothing: antialiased;
+ -moz-osx-font-smoothing: grayscale;
+}
+
+*,
+*::before,
+*::after {
+ box-sizing: border-box;
+}
+
+body {
+ margin: 0;
+ background: #f5f7fb;
+}
+
+button {
+ font-family: inherit;
+}
+
+a {
+ color: inherit;
+}
diff --git a/site-builder/src/main.tsx b/site-builder/src/main.tsx
new file mode 100644
index 00000000..25e7a34e
--- /dev/null
+++ b/site-builder/src/main.tsx
@@ -0,0 +1,13 @@
+import { StrictMode } from 'react';
+import { createRoot } from 'react-dom/client';
+import { BrowserRouter } from 'react-router-dom';
+import './index.css';
+import App from './App.tsx';
+
+createRoot(document.getElementById('root')!).render(
+
+
+
+
+ ,
+);
diff --git a/site-builder/src/pages/dashboard/SiteDashboard.tsx b/site-builder/src/pages/dashboard/SiteDashboard.tsx
new file mode 100644
index 00000000..c2ca9a0c
--- /dev/null
+++ b/site-builder/src/pages/dashboard/SiteDashboard.tsx
@@ -0,0 +1,56 @@
+import { useEffect, useState } from 'react';
+import { Loader2 } from 'lucide-react';
+import { builderApi } from '../../api/builder.api';
+import type { SiteBlueprint } from '../../types/siteBuilder';
+import { Card } from '../../components/common/Card';
+
+export function SiteDashboard() {
+ const [blueprints, setBlueprints] = useState([]);
+ const [loading, setLoading] = useState(false);
+ const [error, setError] = useState();
+
+ useEffect(() => {
+ const fetchData = async () => {
+ setLoading(true);
+ try {
+ const data = await builderApi.listBlueprints();
+ setBlueprints(data);
+ } catch (err) {
+ setError(err instanceof Error ? err.message : 'Unable to load blueprints');
+ } finally {
+ setLoading(false);
+ }
+ };
+ fetchData();
+ }, []);
+
+ return (
+
+ {loading && (
+
+ Loading blueprints…
+
+ )}
+
+ {error && {error}
}
+
+ {!loading && !blueprints.length && (
+ You haven’t generated any sites yet. Launch the wizard to create your first one.
+ )}
+
+
+
+ );
+}
+
+
diff --git a/site-builder/src/pages/preview/PreviewCanvas.tsx b/site-builder/src/pages/preview/PreviewCanvas.tsx
new file mode 100644
index 00000000..5f62322c
--- /dev/null
+++ b/site-builder/src/pages/preview/PreviewCanvas.tsx
@@ -0,0 +1,291 @@
+import { useMemo } from 'react';
+import {
+ FeatureGridBlock,
+ HeroBlock,
+ MarketingTemplate,
+ StatsPanel,
+ type FeatureGridBlockProps,
+ type StatItem,
+} from '@shared';
+import { useSiteDefinitionStore } from '../../state/siteDefinitionStore';
+import type { PageBlock, PageBlueprint, SiteStructure } from '../../types/siteBuilder';
+
+type StructuredContent = Record & {
+ items?: unknown[];
+ eyebrow?: string;
+ ctaLabel?: string;
+ supportingCopy?: string;
+ columns?: number;
+};
+
+export function PreviewCanvas() {
+ const { structure, pages, selectedSlug, selectPage } = useSiteDefinitionStore();
+
+ const page = useMemo(() => {
+ if (structure?.pages?.length) {
+ return structure.pages.find((p) => p.slug === selectedSlug) ?? structure.pages[0];
+ }
+ return pages.find((p) => p.slug === selectedSlug) ?? pages[0];
+ }, [structure, pages, selectedSlug]);
+
+ if (!structure && !pages.length) {
+ return (
+
+
Generate a blueprint to see live previews of every page.
+
+ );
+ }
+
+ const navItems = structure?.site?.primary_navigation ?? pages.map((p) => p.slug);
+ const blocks = getBlocks(page);
+ const heroBlock = blocks.find((block) => normalizeType(block.type) === 'hero');
+ const contentBlocks = heroBlock ? blocks.filter((block) => block !== heroBlock) : blocks;
+
+ const heroSection =
+ heroBlock || page
+ ? renderBlock(heroBlock ?? buildFallbackHero(page, structure))
+ : null;
+
+ const sectionNodes =
+ contentBlocks.length > 0
+ ? contentBlocks.map((block, index) => {renderBlock(block)}
)
+ : buildFallbackSections(page);
+
+ const sidebar = (
+
+
Page objective
+
{page?.objective ?? 'Launch a high-converting page'}
+
+ {buildSidebarInsights(page, structure).map((insight) => (
+ -
+ {insight.label}
+ {insight.value}
+
+ ))}
+
+
+ );
+
+ return (
+
+
+ {navItems?.map((slug) => (
+
+ ))}
+
+
+
+
+ );
+}
+
+function getBlocks(
+ page: (SiteStructure['pages'][number] & { blocks_json?: PageBlock[] }) | PageBlueprint | undefined,
+) {
+ if (!page) return [];
+ const fromStructure = (page as { blocks?: PageBlock[] }).blocks;
+ if (Array.isArray(fromStructure)) return fromStructure;
+ const fromBlueprint = (page as PageBlueprint).blocks_json;
+ return Array.isArray(fromBlueprint) ? fromBlueprint : [];
+}
+
+function renderBlock(block?: PageBlock) {
+ if (!block) return null;
+ const type = normalizeType(block.type);
+ const structuredContent = extractStructuredContent(block);
+ const listContent = extractListContent(block, structuredContent);
+
+ if (type === 'hero') {
+ return (
+ 0 ? (
+
+ {listContent.map((item) => (
+ - {String(item)}
+ ))}
+
+ ) : undefined
+ }
+ />
+ );
+ }
+
+ if (type === 'feature-grid' || type === 'features' || type === 'value-props') {
+ const features = toFeatureList(listContent, structuredContent.items);
+ const columns = normalizeColumns(structuredContent.columns, features.length);
+ return ;
+ }
+
+ if (type === 'stats' || type === 'metrics') {
+ const stats = toStatItems(listContent, structuredContent.items, block);
+ if (!stats.length) return defaultBlock(block);
+ return ;
+ }
+
+ return defaultBlock(block);
+}
+
+function defaultBlock(block: PageBlock) {
+ return (
+
+ {block.heading &&
{block.heading}
}
+ {block.subheading &&
{block.subheading}
}
+ {Array.isArray(block.content) && (
+
+ {block.content.map((item) => (
+ - {item}
+ ))}
+
+ )}
+
+ );
+}
+
+function normalizeType(type?: string) {
+ return (type ?? '').toLowerCase();
+}
+
+function extractStructuredContent(block: PageBlock): StructuredContent {
+ if (Array.isArray(block.content)) {
+ return {};
+ }
+ return (block.content ?? {}) as StructuredContent;
+}
+
+function extractListContent(block: PageBlock, structuredContent: StructuredContent): unknown[] {
+ if (Array.isArray(block.content)) {
+ return block.content;
+ }
+ if (Array.isArray(structuredContent.items)) {
+ return structuredContent.items;
+ }
+ return [];
+}
+
+function toFeatureList(listItems: unknown[], structuredItems?: unknown[]): FeatureGridBlockProps['features'] {
+ const source = structuredItems && Array.isArray(structuredItems) && structuredItems.length > 0 ? structuredItems : listItems;
+ return source.map((item) => {
+ if (typeof item === 'string') {
+ return { title: item };
+ }
+ if (typeof item === 'object' && item) {
+ const record = item as Record;
+ return {
+ title: String(record.title ?? record.heading ?? 'Feature'),
+ description: record.description ? String(record.description) : undefined,
+ icon: record.icon ? String(record.icon) : undefined,
+ };
+ }
+ return { title: String(item) };
+ });
+}
+
+function toStatItems(
+ listItems: unknown[],
+ structuredItems: unknown[] | undefined,
+ block: PageBlock,
+): StatItem[] {
+ const source = structuredItems && Array.isArray(structuredItems) && structuredItems.length > 0 ? structuredItems : listItems;
+ return source
+ .map((item, index) => {
+ if (typeof item === 'string') {
+ return {
+ label: block.heading ?? `Metric ${index + 1}`,
+ value: item,
+ };
+ }
+ if (typeof item === 'object' && item) {
+ const record = item as Record;
+ const label = record.label ?? record.title ?? `Metric ${index + 1}`;
+ const value = record.value ?? record.metric ?? record.score;
+ if (!value) return null;
+ return {
+ label: String(label),
+ value: String(value),
+ description: record.description ? String(record.description) : undefined,
+ };
+ }
+ return null;
+ })
+ .filter((stat): stat is StatItem => Boolean(stat));
+}
+
+function normalizeColumns(
+ candidate: StructuredContent['columns'],
+ featureCount: number,
+): FeatureGridBlockProps['columns'] {
+ const inferred = typeof candidate === 'number' ? candidate : featureCount >= 4 ? 4 : featureCount === 2 ? 2 : 3;
+ if (inferred <= 2) return 2;
+ if (inferred >= 4) return 4;
+ return 3;
+}
+
+function buildFallbackHero(
+ page: SiteStructure['pages'][number] | PageBlueprint | undefined,
+ structure: SiteStructure | undefined,
+): PageBlock {
+ return {
+ type: 'hero',
+ heading: page?.title ?? 'Site Builder preview',
+ subheading: structure?.site?.hero_message ?? 'Preview updates as the AI hydrates your blueprint.',
+ content: Array.isArray(structure?.site?.secondary_navigation) ? structure?.site?.secondary_navigation : [],
+ };
+}
+
+function buildFallbackSections(page: SiteStructure['pages'][number] | PageBlueprint | undefined) {
+ return [
+ ,
+ ,
+ ];
+}
+
+function buildSidebarInsights(
+ page: SiteStructure['pages'][number] | PageBlueprint | undefined,
+ structure: SiteStructure | undefined,
+) {
+ return [
+ {
+ label: 'Primary CTA',
+ value: page?.primary_cta ?? 'Book a demo',
+ },
+ {
+ label: 'Tone',
+ value: structure?.site?.tone ?? 'Confident & clear',
+ },
+ {
+ label: 'Status',
+ value: page?.status ?? 'Draft',
+ },
+ ];
+}
+
+
+
diff --git a/site-builder/src/pages/wizard/WizardPage.tsx b/site-builder/src/pages/wizard/WizardPage.tsx
new file mode 100644
index 00000000..56787162
--- /dev/null
+++ b/site-builder/src/pages/wizard/WizardPage.tsx
@@ -0,0 +1,119 @@
+import { useMemo } from 'react';
+import { Loader2, PlayCircle, RefreshCw } from 'lucide-react';
+import { useBuilderStore } from '../../state/builderStore';
+import { useSiteDefinitionStore } from '../../state/siteDefinitionStore';
+import { BusinessDetailsStep } from './steps/BusinessDetailsStep';
+import { BriefStep } from './steps/BriefStep';
+import { ObjectivesStep } from './steps/ObjectivesStep';
+import { StyleStep } from './steps/StyleStep';
+import { Card } from '../../components/common/Card';
+
+const stepTitles = ['Business', 'Brief', 'Objectives', 'Style'];
+
+export function WizardPage() {
+ const {
+ form,
+ currentStep,
+ setField,
+ updateStyle,
+ addObjective,
+ removeObjective,
+ nextStep,
+ previousStep,
+ setStep,
+ submitWizard,
+ isSubmitting,
+ error,
+ activeBlueprint,
+ refreshPages,
+ } = useBuilderStore();
+ const { structure } = useSiteDefinitionStore();
+
+ const stepComponents = useMemo(
+ () => [
+ ,
+ ,
+ ,
+ ,
+ ],
+ [form, setField, addObjective, removeObjective, updateStyle],
+ );
+
+ return (
+
+
+
+ {stepTitles.map((title, idx) => (
+
+ ))}
+
+
+ {stepComponents[currentStep]}
+
+
+
+ {currentStep < stepComponents.length - 1 ? (
+
+ ) : (
+
+ )}
+
+
+ {error && {error}
}
+
+
+ {activeBlueprint && (
+
refreshPages(activeBlueprint.id)} disabled={isSubmitting}>
+
+ Sync pages
+
+ }
+ >
+
+
+ Status
+ {activeBlueprint.status}
+
+
+ Structure
+ {structure?.pages?.length ?? 0} pages
+
+
+
+ )}
+
+ );
+}
+
+
diff --git a/site-builder/src/pages/wizard/steps/BriefStep.tsx b/site-builder/src/pages/wizard/steps/BriefStep.tsx
new file mode 100644
index 00000000..d29af11e
--- /dev/null
+++ b/site-builder/src/pages/wizard/steps/BriefStep.tsx
@@ -0,0 +1,28 @@
+import type { BuilderFormData } from '../../../types/siteBuilder';
+import { Card } from '../../../components/common/Card';
+
+interface Props {
+ data: BuilderFormData;
+ onChange: (key: K, value: BuilderFormData[K]) => void;
+}
+
+export function BriefStep({ data, onChange }: Props) {
+ return (
+
+
+
+ );
+}
+
+
diff --git a/site-builder/src/pages/wizard/steps/BusinessDetailsStep.tsx b/site-builder/src/pages/wizard/steps/BusinessDetailsStep.tsx
new file mode 100644
index 00000000..20a369a4
--- /dev/null
+++ b/site-builder/src/pages/wizard/steps/BusinessDetailsStep.tsx
@@ -0,0 +1,94 @@
+import type { BuilderFormData } from '../../../types/siteBuilder';
+import { Card } from '../../../components/common/Card';
+
+interface Props {
+ data: BuilderFormData;
+ onChange: (key: K, value: BuilderFormData[K]) => void;
+}
+
+export function BusinessDetailsStep({ data, onChange }: Props) {
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
+
+
diff --git a/site-builder/src/pages/wizard/steps/ObjectivesStep.tsx b/site-builder/src/pages/wizard/steps/ObjectivesStep.tsx
new file mode 100644
index 00000000..53ab33a5
--- /dev/null
+++ b/site-builder/src/pages/wizard/steps/ObjectivesStep.tsx
@@ -0,0 +1,52 @@
+import { useState } from 'react';
+import type { BuilderFormData } from '../../../types/siteBuilder';
+import { Card } from '../../../components/common/Card';
+
+interface Props {
+ data: BuilderFormData;
+ addObjective: (value: string) => void;
+ removeObjective: (index: number) => void;
+}
+
+export function ObjectivesStep({ data, addObjective, removeObjective }: Props) {
+ const [value, setValue] = useState('');
+
+ const handleAdd = () => {
+ const trimmed = value.trim();
+ if (!trimmed) return;
+ addObjective(trimmed);
+ setValue('');
+ };
+
+ return (
+
+
+ {data.objectives.map((objective, idx) => (
+
+ {objective}
+
+
+ ))}
+
+
+
+ setValue(event.target.value)}
+ />
+
+
+
+ );
+}
+
+
diff --git a/site-builder/src/pages/wizard/steps/StyleStep.tsx b/site-builder/src/pages/wizard/steps/StyleStep.tsx
new file mode 100644
index 00000000..abe12c00
--- /dev/null
+++ b/site-builder/src/pages/wizard/steps/StyleStep.tsx
@@ -0,0 +1,74 @@
+import type { StylePreferences } from '../../../types/siteBuilder';
+import { Card } from '../../../components/common/Card';
+
+interface Props {
+ style: StylePreferences;
+ onChange: (partial: Partial) => void;
+}
+
+const palettes = [
+ 'Minimal monochrome with bright accent',
+ 'Rich jewel tones with high contrast',
+ 'Soft gradients and glassmorphism',
+ 'Playful pastel palette',
+];
+
+const typographyOptions = [
+ 'Modern sans-serif for headings, serif body text',
+ 'Editorial serif across the site',
+ 'Geometric sans with tight tracking',
+ 'Rounded fonts with friendly tone',
+];
+
+export function StyleStep({ style, onChange }: Props) {
+ return (
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
+
+
diff --git a/site-builder/src/state/builderStore.ts b/site-builder/src/state/builderStore.ts
new file mode 100644
index 00000000..6df34316
--- /dev/null
+++ b/site-builder/src/state/builderStore.ts
@@ -0,0 +1,156 @@
+import { create } from 'zustand';
+import { builderApi } from '../api/builder.api';
+import type {
+ BuilderFormData,
+ PageBlueprint,
+ SiteBlueprint,
+ StylePreferences,
+} from '../types/siteBuilder';
+import { useSiteDefinitionStore } from './siteDefinitionStore';
+
+const defaultStyle: StylePreferences = {
+ palette: 'Vibrant modern palette with rich accent color',
+ typography: 'Sans-serif display for headings, humanist body font',
+ personality: 'Confident, energetic, optimistic',
+ heroImagery: 'Real people interacting with the product/service',
+};
+
+const defaultForm: BuilderFormData = {
+ siteId: null,
+ sectorId: null,
+ siteName: '',
+ businessType: '',
+ industry: '',
+ targetAudience: '',
+ hostingType: 'igny8_sites',
+ businessBrief: '',
+ objectives: ['Launch a conversion-focused marketing site'],
+ style: defaultStyle,
+};
+
+interface BuilderState {
+ form: BuilderFormData;
+ currentStep: number;
+ isSubmitting: boolean;
+ error?: string;
+ activeBlueprint?: SiteBlueprint;
+ pages: PageBlueprint[];
+ setField: (key: K, value: BuilderFormData[K]) => void;
+ updateStyle: (partial: Partial) => void;
+ addObjective: (value: string) => void;
+ removeObjective: (index: number) => void;
+ setStep: (step: number) => void;
+ nextStep: () => void;
+ previousStep: () => void;
+ reset: () => void;
+ submitWizard: () => Promise;
+ refreshPages: (blueprintId: number) => Promise;
+}
+
+export const useBuilderStore = create((set, get) => ({
+ form: defaultForm,
+ currentStep: 0,
+ isSubmitting: false,
+ pages: [],
+
+ setField: (key, value) =>
+ set((state) => ({
+ form: { ...state.form, [key]: value },
+ })),
+
+ updateStyle: (partial) =>
+ set((state) => ({
+ form: { ...state.form, style: { ...state.form.style, ...partial } },
+ })),
+
+ addObjective: (value) =>
+ set((state) => ({
+ form: { ...state.form, objectives: [...state.form.objectives, value] },
+ })),
+
+ removeObjective: (index) =>
+ set((state) => ({
+ form: {
+ ...state.form,
+ objectives: state.form.objectives.filter((_, idx) => idx !== index),
+ },
+ })),
+
+ setStep: (step) => set({ currentStep: step }),
+
+ nextStep: () =>
+ set((state) => ({
+ currentStep: Math.min(state.currentStep + 1, 3),
+ })),
+
+ previousStep: () =>
+ set((state) => ({
+ currentStep: Math.max(state.currentStep - 1, 0),
+ })),
+
+ reset: () =>
+ set({
+ form: defaultForm,
+ currentStep: 0,
+ isSubmitting: false,
+ error: undefined,
+ activeBlueprint: undefined,
+ pages: [],
+ }),
+
+ submitWizard: async () => {
+ const { form } = get();
+ if (!form.siteId || !form.sectorId) {
+ set({ error: 'Site and sector are required to generate a blueprint.' });
+ return;
+ }
+
+ set({ isSubmitting: true, error: undefined });
+ try {
+ const payload = {
+ name: form.siteName || `Site Blueprint (${form.industry || 'New'})`,
+ description: `${form.businessType} for ${form.targetAudience}`,
+ site_id: form.siteId,
+ sector_id: form.sectorId,
+ hosting_type: form.hostingType,
+ config_json: {
+ business_type: form.businessType,
+ industry: form.industry,
+ target_audience: form.targetAudience,
+ },
+ };
+
+ const blueprint = await builderApi.createBlueprint(payload);
+ set({ activeBlueprint: blueprint });
+
+ const generation = await builderApi.generateStructure(blueprint.id, {
+ business_brief: form.businessBrief,
+ objectives: form.objectives,
+ style: form.style,
+ metadata: { targetAudience: form.targetAudience },
+ });
+
+ if (generation?.structure) {
+ useSiteDefinitionStore.getState().setStructure(generation.structure);
+ }
+
+ await get().refreshPages(blueprint.id);
+ } catch (error) {
+ set({ error: error instanceof Error ? error.message : 'Unexpected error' });
+ } finally {
+ set({ isSubmitting: false });
+ }
+ },
+
+ refreshPages: async (blueprintId: number) => {
+ try {
+ const pages = await builderApi.listPages(blueprintId);
+ set({ pages });
+ useSiteDefinitionStore.getState().setPages(pages);
+ } catch (error) {
+ set({ error: error instanceof Error ? error.message : 'Unable to load pages' });
+ }
+ },
+}));
+
+
diff --git a/site-builder/src/state/siteDefinitionStore.ts b/site-builder/src/state/siteDefinitionStore.ts
new file mode 100644
index 00000000..735b455d
--- /dev/null
+++ b/site-builder/src/state/siteDefinitionStore.ts
@@ -0,0 +1,28 @@
+import { create } from 'zustand';
+import type { PageBlueprint, SiteStructure } from '../types/siteBuilder';
+
+interface SiteDefinitionState {
+ structure?: SiteStructure;
+ pages: PageBlueprint[];
+ selectedSlug?: string;
+ setStructure: (structure: SiteStructure) => void;
+ setPages: (pages: PageBlueprint[]) => void;
+ selectPage: (slug: string) => void;
+}
+
+export const useSiteDefinitionStore = create((set) => ({
+ pages: [],
+ setStructure: (structure) =>
+ set({
+ structure,
+ selectedSlug: structure.pages?.[0]?.slug,
+ }),
+ setPages: (pages) =>
+ set((state) => ({
+ pages,
+ selectedSlug: state.selectedSlug ?? pages[0]?.slug,
+ })),
+ selectPage: (slug) => set({ selectedSlug: slug }),
+}));
+
+
diff --git a/site-builder/src/types/siteBuilder.ts b/site-builder/src/types/siteBuilder.ts
new file mode 100644
index 00000000..931ec479
--- /dev/null
+++ b/site-builder/src/types/siteBuilder.ts
@@ -0,0 +1,87 @@
+export type HostingType = 'igny8_sites' | 'wordpress' | 'shopify' | 'multi';
+
+export interface StylePreferences {
+ palette: string;
+ typography: string;
+ personality: string;
+ heroImagery: string;
+}
+
+export interface BuilderFormData {
+ siteId: number | null;
+ sectorId: number | null;
+ siteName: string;
+ businessType: string;
+ industry: string;
+ targetAudience: string;
+ hostingType: HostingType;
+ businessBrief: string;
+ objectives: string[];
+ style: StylePreferences;
+}
+
+export interface SiteBlueprint {
+ id: number;
+ name: string;
+ description?: string;
+ status: 'draft' | 'generating' | 'ready' | 'deployed';
+ hosting_type: HostingType;
+ config_json: Record;
+ structure_json: SiteStructure | null;
+ created_at: string;
+ updated_at: string;
+}
+
+export interface PageBlueprint {
+ id: number;
+ site_blueprint: number;
+ slug: string;
+ title: string;
+ type: string;
+ status: string;
+ order: number;
+ blocks_json: PageBlock[];
+}
+
+export interface PageBlock {
+ type: string;
+ heading?: string;
+ subheading?: string;
+ layout?: string;
+ content?: string[] | Record;
+}
+
+export interface SiteStructure {
+ site?: {
+ name?: string;
+ primary_navigation?: string[];
+ secondary_navigation?: string[];
+ hero_message?: string;
+ tone?: string;
+ };
+ pages: Array<{
+ slug: string;
+ title: string;
+ type: string;
+ status?: string;
+ objective?: string;
+ primary_cta?: string;
+ blocks?: PageBlock[];
+ }>;
+}
+
+export interface ApiListResponse {
+ count?: number;
+ next?: string | null;
+ previous?: string | null;
+ results?: T[];
+ data?: T[] | T;
+}
+
+export interface ApiError {
+ message?: string;
+ error?: string;
+ detail?: string;
+}
+
+
diff --git a/site-builder/tsconfig.app.json b/site-builder/tsconfig.app.json
new file mode 100644
index 00000000..a6aa902d
--- /dev/null
+++ b/site-builder/tsconfig.app.json
@@ -0,0 +1,31 @@
+{
+ "compilerOptions": {
+ "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
+ "target": "ES2022",
+ "useDefineForClassFields": true,
+ "lib": ["ES2022", "DOM", "DOM.Iterable"],
+ "module": "ESNext",
+ "types": ["vite/client"],
+ "skipLibCheck": true,
+
+ /* Bundler mode */
+ "moduleResolution": "bundler",
+ "allowImportingTsExtensions": true,
+ "verbatimModuleSyntax": true,
+ "moduleDetection": "force",
+ "noEmit": true,
+ "jsx": "react-jsx",
+ "baseUrl": ".",
+ "paths": {
+ "@shared/*": ["../frontend/src/components/shared/*"]
+ },
+ /* Linting */
+ "strict": true,
+ "noUnusedLocals": true,
+ "noUnusedParameters": true,
+ "erasableSyntaxOnly": true,
+ "noFallthroughCasesInSwitch": true,
+ "noUncheckedSideEffectImports": true
+ },
+ "include": ["src"]
+}
diff --git a/site-builder/tsconfig.json b/site-builder/tsconfig.json
new file mode 100644
index 00000000..1ffef600
--- /dev/null
+++ b/site-builder/tsconfig.json
@@ -0,0 +1,7 @@
+{
+ "files": [],
+ "references": [
+ { "path": "./tsconfig.app.json" },
+ { "path": "./tsconfig.node.json" }
+ ]
+}
diff --git a/site-builder/tsconfig.node.json b/site-builder/tsconfig.node.json
new file mode 100644
index 00000000..8a67f62f
--- /dev/null
+++ b/site-builder/tsconfig.node.json
@@ -0,0 +1,26 @@
+{
+ "compilerOptions": {
+ "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo",
+ "target": "ES2023",
+ "lib": ["ES2023"],
+ "module": "ESNext",
+ "types": ["node"],
+ "skipLibCheck": true,
+
+ /* Bundler mode */
+ "moduleResolution": "bundler",
+ "allowImportingTsExtensions": true,
+ "verbatimModuleSyntax": true,
+ "moduleDetection": "force",
+ "noEmit": true,
+
+ /* Linting */
+ "strict": true,
+ "noUnusedLocals": true,
+ "noUnusedParameters": true,
+ "erasableSyntaxOnly": true,
+ "noFallthroughCasesInSwitch": true,
+ "noUncheckedSideEffectImports": true
+ },
+ "include": ["vite.config.ts"]
+}
diff --git a/site-builder/vite.config.ts b/site-builder/vite.config.ts
new file mode 100644
index 00000000..3d1b5ba4
--- /dev/null
+++ b/site-builder/vite.config.ts
@@ -0,0 +1,31 @@
+import fs from 'node:fs';
+import path from 'node:path';
+import { fileURLToPath } from 'node:url';
+import { defineConfig } from 'vite';
+import react from '@vitejs/plugin-react';
+
+const __filename = fileURLToPath(import.meta.url);
+const __dirname = path.dirname(__filename);
+const sharedPathCandidates = [
+ path.resolve(__dirname, '../frontend/src/components/shared'),
+ path.resolve(__dirname, '../../frontend/src/components/shared'),
+ '/frontend/src/components/shared',
+];
+const sharedComponentsPath = sharedPathCandidates.find((candidate) => fs.existsSync(candidate)) ?? sharedPathCandidates[0];
+
+export default defineConfig({
+ plugins: [react()],
+ resolve: {
+ alias: {
+ '@shared': sharedComponentsPath,
+ },
+ },
+ server: {
+ host: '0.0.0.0',
+ port: 5175,
+ allowedHosts: ['builder.igny8.com'],
+ fs: {
+ allow: [path.resolve(__dirname, '..'), sharedComponentsPath],
+ },
+ },
+});