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 &&
{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.

+ )} + +
    + {blueprints.map((bp) => ( +
  • +
    + {bp.name} + {bp.description} +
    + {bp.status} +
  • + ))} +
+
+ ); +} + + 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 ( + +