commit 0fee703b842b99fd34b3c73aa313919bac3d635f Author: beabigegg Date: Sun Aug 17 15:26:44 2025 +0800 back diff --git a/.env b/.env new file mode 100644 index 0000000..0d5f07e --- /dev/null +++ b/.env @@ -0,0 +1,19 @@ +# Dify API Base URL (Common for all apps) +DIFY_API_BASE_URL="https://dify.theaken.com/v1" + +# --- Dify API Keys for Specific Apps --- +DIFY_TRANSLATOR_API_KEY="app-YOPrF2ro5fshzMkCZviIuUJd" +DIFY_SUMMARIZER_API_KEY="app-oFptWFRlSgvwhJ8DzZKN08a0" +DIFY_ACTION_EXTRACTOR_API_KEY="app-UHU5IrVcwE0nVvgzubpGRqym" +DIFY_STT_API_KEY="app-xQeSipaQecs0cuKeLvYDaRsu" + +# Celery Configuration +CELERY_BROKER_URL="redis://localhost:6379/0" +CELERY_RESULT_BACKEND="redis://localhost:6379/0" + +# Flask App Configuration +FLASK_RUN_PORT=12000 + +# Database and JWT Configuration +DATABASE_URL="mysql+pymysql://A060:WLeSCi0yhtc7@mysql.theaken.com:33306/db_A060" +JWT_SECRET_KEY="your-super-secret-key-that-no-one-should-know" diff --git a/__pycache__/action_item_routes.cpython-312.pyc b/__pycache__/action_item_routes.cpython-312.pyc new file mode 100644 index 0000000..7d092ad Binary files /dev/null and b/__pycache__/action_item_routes.cpython-312.pyc differ diff --git a/__pycache__/ai_routes.cpython-312.pyc b/__pycache__/ai_routes.cpython-312.pyc new file mode 100644 index 0000000..acc5295 Binary files /dev/null and b/__pycache__/ai_routes.cpython-312.pyc differ diff --git a/__pycache__/api_routes.cpython-312.pyc b/__pycache__/api_routes.cpython-312.pyc new file mode 100644 index 0000000..4906f3e Binary files /dev/null and b/__pycache__/api_routes.cpython-312.pyc differ diff --git a/__pycache__/app.cpython-312.pyc b/__pycache__/app.cpython-312.pyc new file mode 100644 index 0000000..a409ade Binary files /dev/null and b/__pycache__/app.cpython-312.pyc differ diff --git a/__pycache__/celery_app.cpython-312.pyc b/__pycache__/celery_app.cpython-312.pyc new file mode 100644 index 0000000..a54ff01 Binary files /dev/null and b/__pycache__/celery_app.cpython-312.pyc differ diff --git a/__pycache__/celery_worker.cpython-312.pyc b/__pycache__/celery_worker.cpython-312.pyc new file mode 100644 index 0000000..3e78be8 Binary files /dev/null and b/__pycache__/celery_worker.cpython-312.pyc differ diff --git a/__pycache__/models.cpython-312.pyc b/__pycache__/models.cpython-312.pyc new file mode 100644 index 0000000..75ffd33 Binary files /dev/null and b/__pycache__/models.cpython-312.pyc differ diff --git a/__pycache__/tasks.cpython-312.pyc b/__pycache__/tasks.cpython-312.pyc new file mode 100644 index 0000000..fe3da5c Binary files /dev/null and b/__pycache__/tasks.cpython-312.pyc differ diff --git a/action_item_routes.py b/action_item_routes.py new file mode 100644 index 0000000..533a0b9 --- /dev/null +++ b/action_item_routes.py @@ -0,0 +1,112 @@ +# action_item_routes.py +from flask import Blueprint, request, jsonify +from flask_jwt_extended import jwt_required +from datetime import date +from models import db, User, Meeting, ActionItem + +action_bp = Blueprint("action_bp", __name__, url_prefix="/api") + +def _parse_date(s: str | None): + if not s: + return None + try: + return date.fromisoformat(str(s)[:10]) + except Exception: + return None + +def _resolve_owner_id(owner_val: str | None): + """把 'owner'(使用者名稱字串) 轉到 ms_users.id,查不到就回 None。""" + if not owner_val: + return None + owner_val = str(owner_val).strip() + if not owner_val: + return None + user = User.query.filter_by(username=owner_val).first() + return user.id if user else None + +@action_bp.post("/action-items") +@jwt_required() +def create_action_item(): + """ + 建立單筆代辦(會議詳情頁用) + 允許欄位:meeting_id(必) / item(或 context) / action(必) / owner_id or owner(使用者名) / due_date or duedate / status + """ + data = request.get_json(force=True) or {} + meeting_id = data.get("meeting_id") + if not meeting_id: + return jsonify({"error": "meeting_id is required"}), 400 + meeting = Meeting.query.get(meeting_id) + if not meeting: + return jsonify({"error": "meeting not found"}), 404 + + action_text = (data.get("action") or "").strip() + if not action_text: + return jsonify({"error": "action is required"}), 400 + + item_text = (data.get("item") or data.get("context") or "").strip() or None + owner_id = data.get("owner_id") + if owner_id is None: + owner_id = _resolve_owner_id(data.get("owner")) + due = _parse_date(data.get("due_date") or data.get("duedate")) + status = (data.get("status") or "pending").strip() or "pending" + + try: + ai = ActionItem( + meeting_id=meeting_id, + item=item_text, + action=action_text, + owner_id=owner_id, + due_date=due, + status=status, + ) + db.session.add(ai) + db.session.commit() + return jsonify(ai.to_dict()), 201 + except Exception as e: + db.session.rollback() + return jsonify({"error": f"create failed: {e}"}), 400 + +@action_bp.post("/meetings//action-items/batch") +@jwt_required() +def batch_create_action_items(meeting_id: int): + """ + 批次建立代辦(AI 預覽 → 一鍵儲存) + Request body: { "items": [ {item/context, action*, owner/owner_id, due_date/duedate}, ... ] } + """ + payload = request.get_json(force=True) or {} + items = payload.get("items") or [] + if not isinstance(items, list) or not items: + return jsonify({"error": "items must be a non-empty array"}), 400 + + meeting = Meeting.query.get(meeting_id) + if not meeting: + return jsonify({"error": "meeting not found"}), 404 + + created = [] + try: + for r in items: + action_text = (r.get("action") or "").strip() + if not action_text: + continue # 沒有 action 的略過 + + item_text = (r.get("item") or r.get("context") or "").strip() or None + owner_id = r.get("owner_id") + if owner_id is None: + owner_id = _resolve_owner_id(r.get("owner")) + due = _parse_date(r.get("due_date") or r.get("duedate")) + + ai = ActionItem( + meeting_id=meeting_id, + item=item_text, + action=action_text, + owner_id=owner_id, + due_date=due, + status="pending", + ) + db.session.add(ai) + created.append(ai) + db.session.commit() + return jsonify([c.to_dict() for c in created]), 201 + except Exception as e: + db.session.rollback() + return jsonify({"error": f"batch create failed: {e}"}), 400 diff --git a/ai_routes.py b/ai_routes.py new file mode 100644 index 0000000..392575c --- /dev/null +++ b/ai_routes.py @@ -0,0 +1,40 @@ +# ai_routes.py +from flask import Blueprint, request, jsonify +from flask_jwt_extended import jwt_required, get_jwt_identity +from services.dify_client import translate_text as _translate_text, summarize_text as _summarize_text, extract_action_items as _extract_action_items + +ai_bp = Blueprint("ai_bp", __name__, url_prefix="/api") + +@ai_bp.post("/translate/text") +@jwt_required() +def translate_text_api(): + data = request.get_json(force=True) or {} + text = (data.get("text") or "").strip() + target = (data.get("target_lang") or "繁體中文").strip() + if not text: + return jsonify({"error": "text is required"}), 400 + user_id = str(get_jwt_identity() or "user") + translated = _translate_text(text, target, user_id=user_id) + return jsonify({"translated": translated}) + +@ai_bp.post("/summarize/text") +@jwt_required() +def summarize_text_api(): + data = request.get_json(force=True) or {} + text = (data.get("text") or "").strip() + if not text: + return jsonify({"error": "text is required"}), 400 + user_id = str(get_jwt_identity() or "user") + summary = _summarize_text(text, user_id=user_id) + return jsonify({"summary": summary}) + +@ai_bp.post("/action-items/preview") +@jwt_required() +def preview_action_items_api(): + data = request.get_json(force=True) or {} + text = (data.get("text") or "").strip() + if not text: + return jsonify({"error": "text is required"}), 400 + user_id = str(get_jwt_identity() or "user") + items = _extract_action_items(text, user_id=user_id) + return jsonify({"items": items}) diff --git a/api_routes.py b/api_routes.py new file mode 100644 index 0000000..63bfc5b --- /dev/null +++ b/api_routes.py @@ -0,0 +1,349 @@ +import os +import uuid +from flask import request, jsonify, send_from_directory, Blueprint, current_app +from flask_jwt_extended import create_access_token, jwt_required, get_jwt_identity, get_jwt +from werkzeug.utils import secure_filename +from datetime import datetime + +from models import db, User, Meeting, ActionItem +from tasks import ( + celery, + extract_audio_task, + transcribe_audio_task, + translate_text_task, + summarize_text_task +) + +api_bp = Blueprint("api_bp", __name__, url_prefix="/api") + +def save_uploaded_file(upload_folder, file_key='file'): + if file_key not in request.files: + return None, (jsonify({'error': 'Request does not contain a file part'}), 400) + file = request.files[file_key] + if file.filename == '': + return None, (jsonify({'error': 'No file selected'}), 400) + if file: + original_filename = secure_filename(file.filename) + file_extension = os.path.splitext(original_filename)[1] + unique_filename = f"{uuid.uuid4()}{file_extension}" + file_path = os.path.join(upload_folder, unique_filename) + file.save(file_path) + return file_path, None + return None, (jsonify({'error': 'Unknown file error'}), 500) + +# --- User Authentication & Admin Routes --- +@api_bp.route('/login', methods=['POST']) +def login(): + data = request.get_json() + user = User.query.filter_by(username=data.get('username')).first() + if user and user.check_password(data.get('password')): + access_token = create_access_token(identity=str(user.id), additional_claims={'role': user.role, 'username': user.username}) + return jsonify(access_token=access_token) + return jsonify({"msg": "Bad username or password"}), 401 + +@api_bp.route('/register', methods=['POST']) +def register(): + """Public endpoint for new user registration.""" + data = request.get_json() + username = data.get('username') + password = data.get('password') + + if not username or not password: + return jsonify({"error": "Username and password are required"}), 400 + + if User.query.filter_by(username=username).first(): + return jsonify({"error": "Username already exists"}), 409 # HTTP 409 Conflict + + try: + new_user = User(username=username, role='user') # Default role is 'user' + new_user.set_password(password) + db.session.add(new_user) + db.session.commit() + return jsonify({"message": "User created successfully"}), 201 # HTTP 201 Created + except Exception as e: + db.session.rollback() + current_app.logger.error(f"Error creating user: {e}") + return jsonify({"error": "An internal error occurred"}), 500 + +@api_bp.route('/admin/users', methods=['GET']) +@jwt_required() +def get_all_users(): + if get_jwt().get('role') != 'admin': + return jsonify({"msg": "Administration rights required"}), 403 + users = User.query.all() + return jsonify([user.to_dict() for user in users]) + +@api_bp.route('/users', methods=['GET']) +@jwt_required() +def get_all_users_for_dropdown(): + """A public endpoint for all logged-in users to fetch a list of users for UI selectors.""" + users = User.query.all() + return jsonify([user.to_dict() for user in users]) + +@api_bp.route('/admin/users', methods=['POST']) +@jwt_required() +def create_user(): + if get_jwt().get('role') != 'admin': + return jsonify({"msg": "Administration rights required"}), 403 + data = request.get_json() + username = data.get('username') + password = data.get('password') + role = data.get('role', 'user') + if not username or not password: + return jsonify({"error": "Username and password are required"}), 400 + if User.query.filter_by(username=username).first(): + return jsonify({"error": "Username already exists"}), 409 + + new_user = User(username=username, role=role) + new_user.set_password(password) + db.session.add(new_user) + db.session.commit() + return jsonify(new_user.to_dict()), 201 + +@api_bp.route('/admin/users/', methods=['DELETE']) +@jwt_required() +def delete_user(user_id): + if get_jwt().get('role') != 'admin': + return jsonify({"msg": "Administration rights required"}), 403 + + # Prevent admin from deleting themselves + if str(user_id) == get_jwt_identity(): + return jsonify({"error": "Admin users cannot delete their own account"}), 400 + + user_to_delete = User.query.get(user_id) + if not user_to_delete: + return jsonify({"error": "User not found"}), 404 + + try: + # Disassociate meetings created by this user + Meeting.query.filter_by(created_by_id=user_id).update({"created_by_id": None}) + + # Disassociate action items owned by this user + ActionItem.query.filter_by(owner_id=user_id).update({"owner_id": None}) + + db.session.delete(user_to_delete) + db.session.commit() + return jsonify({"msg": f"User {user_to_delete.username} has been deleted."}), 200 + except Exception as e: + db.session.rollback() + return jsonify({"error": f"An error occurred: {str(e)}"}), 500 + +@api_bp.route('/admin/users//password', methods=['PUT']) +@jwt_required() +def update_user_password(user_id): + if get_jwt().get('role') != 'admin': + return jsonify({"msg": "Administration rights required"}), 403 + + user_to_update = User.query.get(user_id) + if not user_to_update: + return jsonify({"error": "User not found"}), 404 + + data = request.get_json() + password = data.get('password') + if not password: + return jsonify({"error": "Password is required"}), 400 + + try: + user_to_update.set_password(password) + db.session.commit() + return jsonify({"msg": f"Password for user {user_to_update.username} has been updated."}), 200 + except Exception as e: + db.session.rollback() + current_app.logger.error(f"Error updating password for user {user_id}: {e}") + return jsonify({"error": "An internal error occurred while updating the password"}), 500 + +# --- Meeting Management Routes --- +@api_bp.route('/meetings', methods=['GET', 'POST']) +@jwt_required() +def handle_meetings(): + if request.method == 'POST': + data = request.get_json() + topic = data.get('topic') + meeting_date_str = data.get('meeting_date') + if not topic or not meeting_date_str: + return jsonify({'error': 'Topic and meeting_date are required'}), 400 + try: + meeting_date = datetime.fromisoformat(meeting_date_str).date() + new_meeting = Meeting( + topic=topic, + meeting_date=meeting_date, + created_by_id=get_jwt_identity(), + created_at=datetime.utcnow(), # Explicitly set creation time in UTC + status='In Progress' # Set default status to 'In Progress' + ) + db.session.add(new_meeting) + db.session.commit() + return jsonify(new_meeting.to_dict()), 201 + except Exception as e: + db.session.rollback() + current_app.logger.error(f"Failed to create meeting: {e}") + return jsonify({'error': 'Failed to create meeting due to a database error.'}), 500 + + meetings = Meeting.query.order_by(Meeting.meeting_date.desc()).all() + return jsonify([meeting.to_dict() for meeting in meetings]) + +@api_bp.route('/meetings/', methods=['GET', 'PUT', 'DELETE']) +@jwt_required() +def handle_meeting_detail(meeting_id): + meeting = Meeting.query.get_or_404(meeting_id) + + if request.method == 'PUT': + data = request.get_json() + # Only update fields that are present in the request + if 'topic' in data: + meeting.topic = data.get('topic') + if 'status' in data: + # Security check: only admin or meeting creator can change the status + current_user_id = get_jwt_identity() + is_admin = get_jwt().get('role') == 'admin' + if not is_admin and str(meeting.created_by_id) != str(current_user_id): + return jsonify({"msg": "Only the meeting creator or an admin can change the status."}), 403 + meeting.status = data.get('status') + if 'transcript' in data: + meeting.transcript = data.get('transcript') + if 'summary' in data: + meeting.summary = data.get('summary') + if data.get('meeting_date'): + meeting.meeting_date = datetime.fromisoformat(data['meeting_date']).date() + + db.session.commit() + # Refresh the object to avoid session state issues before serialization + db.session.refresh(meeting) + return jsonify(meeting.to_dict()) + + if request.method == 'DELETE': + if get_jwt().get('role') != 'admin': + return jsonify({"msg": "Administration rights required"}), 403 + db.session.delete(meeting) + db.session.commit() + return jsonify({"msg": "Meeting and associated action items deleted"}), 200 + + # GET request + return jsonify(meeting.to_dict()) + +@api_bp.route('/meetings//summarize', methods=['POST']) +@jwt_required() +def summarize_meeting(meeting_id): + meeting = Meeting.query.get_or_404(meeting_id) + if not meeting.transcript: + return jsonify({'error': 'Meeting has no transcript to summarize.'}), 400 + task = summarize_text_task.delay(meeting_id) + return jsonify({'task_id': task.id, 'status_url': f'/status/{task.id}'}), 202 + +# --- Independent Tool Routes --- +@api_bp.route('/tools/extract_audio', methods=['POST']) +@jwt_required() +def handle_extract_audio(): + input_path, error = save_uploaded_file(current_app.config['UPLOAD_FOLDER']) + if error: return error + output_path = os.path.splitext(input_path)[0] + ".wav" + task = extract_audio_task.delay(input_path, output_path) + return jsonify({'task_id': task.id, 'status_url': f'/status/{task.id}'}), 202 + +@api_bp.route('/tools/transcribe_audio', methods=['POST']) +@jwt_required() +def handle_transcribe_audio(): + input_path, error = save_uploaded_file(current_app.config['UPLOAD_FOLDER']) + if error: return error + # The 'language' parameter is no longer needed for the Dify-based task. + task = transcribe_audio_task.delay(input_path) + return jsonify({'task_id': task.id, 'status_url': f'/status/{task.id}'}), 202 + +@api_bp.route('/tools/translate_text', methods=['POST']) +@jwt_required() +def handle_translate_text(): + data = request.get_json() + text_content = data.get('text') + target_language = data.get('target_language', '繁體中文') + if not text_content: + return jsonify({'error': 'Text content is required'}), 400 + task = translate_text_task.delay(text_content, target_language) + return jsonify({'task_id': task.id, 'status_url': f'/status/{task.id}'}), 202 + +# --- Action Item & Task Status Routes (largely unchanged) --- +@api_bp.route('/meetings//action_items', methods=['GET']) +@jwt_required() +def get_action_items_for_meeting(meeting_id): + action_items = ActionItem.query.filter_by(meeting_id=meeting_id).all() + return jsonify([item.to_dict() for item in action_items]) + +@api_bp.route('/action_items/', methods=['PUT', 'DELETE']) +@jwt_required() +def handle_action_item(item_id): + item = ActionItem.query.get_or_404(item_id) + current_user_id = get_jwt_identity() + current_user_role = get_jwt().get('role') + meeting_owner_id = str(item.meeting.created_by_id) + + is_admin = current_user_role == 'admin' + is_meeting_owner = str(current_user_id) == meeting_owner_id + is_action_owner = str(current_user_id) == str(item.owner_id) + + if request.method == 'PUT': + # Edit Permission: Admin, Meeting Owner, or Action Item Owner + if not (is_admin or is_meeting_owner or is_action_owner): + return jsonify({"msg": "You do not have permission to edit this item."}), 403 + + data = request.get_json() + item.item = data.get('item', item.item) + item.action = data.get('action', item.action) + item.status = data.get('status', item.status) + # Handle owner_id, allowing it to be set to null + if 'owner_id' in data: + item.owner_id = data.get('owner_id') if data.get('owner_id') else None + if data.get('due_date'): + item.due_date = datetime.fromisoformat(data['due_date']).date() if data['due_date'] else None + db.session.commit() + db.session.refresh(item) + return jsonify(item.to_dict()) + + elif request.method == 'DELETE': + # Delete Permission: Admin or Meeting Owner + if not (is_admin or is_meeting_owner): + return jsonify({"msg": "You do not have permission to delete this item."}), 403 + + db.session.delete(item) + db.session.commit() + return jsonify({'msg': 'Action item deleted'}), 200 + +@api_bp.route('/action_items//upload', methods=['POST']) +@jwt_required() +def upload_action_item_attachment(item_id): + item = ActionItem.query.get_or_404(item_id) + + # Basic permission check: only meeting creator or action item owner can upload + meeting_creator_id = item.meeting.created_by_id + current_user_id = get_jwt_identity() + + if str(current_user_id) != str(meeting_creator_id) and str(current_user_id) != str(item.owner_id): + return jsonify({"msg": "Permission denied"}), 403 + + file_path, error = save_uploaded_file(current_app.config['UPLOAD_FOLDER']) + if error: + return error + + # TODO: Consider deleting the old file if it exists + item.attachment_path = os.path.basename(file_path) + db.session.commit() + + return jsonify({'attachment_path': item.attachment_path}), 200 + +@api_bp.route('/status/') +@jwt_required() +def get_task_status(task_id): + task = celery.AsyncResult(task_id) + response_data = {'state': task.state, 'info': task.info if isinstance(task.info, dict) else str(task.info)} + if task.state == 'SUCCESS' and isinstance(task.info, dict) and 'result_path' in task.info: + response_data['info']['download_filename'] = os.path.basename(task.info['result_path']) + return jsonify(response_data) + +@api_bp.route('/task//stop', methods=['POST']) +@jwt_required() +def stop_task(task_id): + celery.control.revoke(task_id, terminate=True) + return jsonify({'status': 'revoked'}), 200 + +@api_bp.route('/download/') +@jwt_required() +def download_file(filename): + return send_from_directory(current_app.config['UPLOAD_FOLDER'], filename, as_attachment=True) diff --git a/app.py b/app.py new file mode 100644 index 0000000..ce65fdc --- /dev/null +++ b/app.py @@ -0,0 +1,95 @@ +import os +import click +from datetime import timedelta +from flask import Flask +from dotenv import load_dotenv +from flask_migrate import Migrate +from flask_jwt_extended import JWTManager +from flask_cors import CORS + +from models import db, bcrypt, User +from celery_app import celery # Import celery instance + +def create_app(): + """Application Factory Pattern""" + load_dotenv() + app = Flask(__name__) + + # --- Configuration --- + app.config.from_mapping( + SQLALCHEMY_DATABASE_URI=os.environ.get('DATABASE_URL'), + JWT_SECRET_KEY=os.environ.get('JWT_SECRET_KEY'), + SQLALCHEMY_TRACK_MODIFICATIONS=False, + JWT_ACCESS_TOKEN_EXPIRES=timedelta(days=3), + CELERY_BROKER_URL=os.environ.get('CELERY_BROKER_URL', 'redis://localhost:6379/0'), + CELERY_RESULT_BACKEND=os.environ.get('CELERY_RESULT_BACKEND', 'redis://localhost:6379/0'), + DIFY_API_BASE_URL=os.environ.get("DIFY_API_BASE_URL"), + DIFY_STT_API_KEY=os.environ.get("DIFY_STT_API_KEY"), + DIFY_TRANSLATOR_API_KEY=os.environ.get("DIFY_TRANSLATOR_API_KEY"), + DIFY_SUMMARIZER_API_KEY=os.environ.get("DIFY_SUMMARIZER_API_KEY"), + DIFY_ACTION_EXTRACTOR_API_KEY=os.environ.get("DIFY_ACTION_EXTRACTOR_API_KEY") + ) + + project_root = os.path.dirname(os.path.abspath(__file__)) + UPLOAD_FOLDER = os.path.join(project_root, 'uploads') + if not os.path.exists(UPLOAD_FOLDER): + os.makedirs(UPLOAD_FOLDER) + app.config['UPLOAD_FOLDER'] = UPLOAD_FOLDER + app.config['MAX_CONTENT_LENGTH'] = 1024 * 1024 * 1024 # 1GB upload limit + + # --- Initialize Extensions --- + db.init_app(app) + bcrypt.init_app(app) + Migrate(app, db) + JWTManager(app) + CORS(app) + + # --- Configure Celery --- + celery.conf.update(app.config) + # This custom Task class ensures tasks run with the Flask app context + class ContextTask(celery.Task): + def __call__(self, *args, **kwargs): + with app.app_context(): + return self.run(*args, **kwargs) + celery.Task = ContextTask + + # --- Import and Register Blueprints --- + from api_routes import api_bp + from ai_routes import ai_bp + from action_item_routes import action_bp + app.register_blueprint(api_bp) + app.register_blueprint(ai_bp) + app.register_blueprint(action_bp) + + # --- Root Route --- + @app.route('/') + def index(): + return "AI Meeting Assistant Backend is running." + + # --- CLI Commands --- + @app.cli.command("create_admin") + @click.argument("username") + @click.argument("password") + def create_admin(username, password): + """Creates a new admin user.""" + with app.app_context(): + try: + if User.query.filter_by(username=username).first(): + print(f"Error: User '{username}' already exists.") + return + admin_user = User(username=username, role='admin') + admin_user.set_password(password) + db.session.add(admin_user) + db.session.commit() + print(f"Admin user '{username}' created successfully.") + except Exception as e: + db.session.rollback() + print(f"An error occurred: {e}") + + return app + +app = create_app() + +if __name__ == '__main__': + port = int(os.environ.get("FLASK_RUN_PORT", 5000)) + app.run(host='0.0.0.0', port=port, debug=True) diff --git a/celery_app.py b/celery_app.py new file mode 100644 index 0000000..2460d11 --- /dev/null +++ b/celery_app.py @@ -0,0 +1,13 @@ +from celery import Celery +from dotenv import load_dotenv + +# Load environment variables from .env file +load_dotenv() + +CELERY_BROKER_URL = 'redis://localhost:6379/0' +CELERY_RESULT_BACKEND = 'redis://localhost:6379/0' + +celery = Celery('tasks', + broker=CELERY_BROKER_URL, + backend=CELERY_RESULT_BACKEND, + include=['tasks']) # Point to the tasks module diff --git a/celery_worker.py b/celery_worker.py new file mode 100644 index 0000000..747359b --- /dev/null +++ b/celery_worker.py @@ -0,0 +1,14 @@ +# This monkey-patching is crucial for eventlet/gevent to work correctly. +# It must be done at the very top, before any other modules are imported. +import eventlet +eventlet.monkey_patch() + +from dotenv import load_dotenv +# Load environment variables BEFORE creating the app +load_dotenv() + +from app import create_app +from celery_app import celery + +app = create_app() +app.app_context().push() diff --git a/frontend/.gitignore b/frontend/.gitignore new file mode 100644 index 0000000..a547bf3 --- /dev/null +++ b/frontend/.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/frontend/README.md b/frontend/README.md new file mode 100644 index 0000000..7059a96 --- /dev/null +++ b/frontend/README.md @@ -0,0 +1,12 @@ +# React + 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/) 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 + +## Expanding the ESLint configuration + +If you are developing a production application, we recommend using TypeScript with type-aware lint rules enabled. Check out the [TS template](https://github.com/vitejs/vite/tree/main/packages/create-vite/template-react-ts) for information on how to integrate TypeScript and [`typescript-eslint`](https://typescript-eslint.io) in your project. diff --git a/frontend/eslint.config.js b/frontend/eslint.config.js new file mode 100644 index 0000000..cee1e2c --- /dev/null +++ b/frontend/eslint.config.js @@ -0,0 +1,29 @@ +import js from '@eslint/js' +import globals from 'globals' +import reactHooks from 'eslint-plugin-react-hooks' +import reactRefresh from 'eslint-plugin-react-refresh' +import { defineConfig, globalIgnores } from 'eslint/config' + +export default defineConfig([ + globalIgnores(['dist']), + { + files: ['**/*.{js,jsx}'], + extends: [ + js.configs.recommended, + reactHooks.configs['recommended-latest'], + reactRefresh.configs.vite, + ], + languageOptions: { + ecmaVersion: 2020, + globals: globals.browser, + parserOptions: { + ecmaVersion: 'latest', + ecmaFeatures: { jsx: true }, + sourceType: 'module', + }, + }, + rules: { + 'no-unused-vars': ['error', { varsIgnorePattern: '^[A-Z_]' }], + }, + }, +]) diff --git a/frontend/index.html b/frontend/index.html new file mode 100644 index 0000000..8a9181c --- /dev/null +++ b/frontend/index.html @@ -0,0 +1,13 @@ + + + + + + + Meeting Assistant + + +
+ + + diff --git a/frontend/npm b/frontend/npm new file mode 100644 index 0000000..e69de29 diff --git a/frontend/package-lock.json b/frontend/package-lock.json new file mode 100644 index 0000000..9894ce1 --- /dev/null +++ b/frontend/package-lock.json @@ -0,0 +1,3805 @@ +{ + "name": "frontend", + "version": "0.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "frontend", + "version": "0.0.0", + "dependencies": { + "@emotion/react": "^11.14.0", + "@emotion/styled": "^11.14.1", + "@mui/icons-material": "^7.3.1", + "@mui/material": "^7.3.1", + "axios": "^1.11.0", + "jwt-decode": "^4.0.0", + "react": "^19.1.1", + "react-dom": "^19.1.1", + "react-router-dom": "^7.8.0" + }, + "devDependencies": { + "@eslint/js": "^9.33.0", + "@types/react": "^19.1.10", + "@types/react-dom": "^19.1.7", + "@vitejs/plugin-react": "^5.0.0", + "eslint": "^9.33.0", + "eslint-plugin-react-hooks": "^5.2.0", + "eslint-plugin-react-refresh": "^0.4.20", + "globals": "^16.3.0", + "vite": "^7.1.2" + } + }, + "node_modules/@ampproject/remapping": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", + "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "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==", + "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.0", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.0.tgz", + "integrity": "sha512-60X7qkglvrap8mn1lh2ebxXdZYtUcpd7gsmy9kLaBJ4i/WdY8PqTSdxyA8qraikqKQK5C1KRBKXqznrVapyNaw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.0.tgz", + "integrity": "sha512-UlLAnTPrFdNGoFtbSXwcGFQBtQZJCNjaN6hQNP3UPvuNXT1i82N26KL3dZeIpNalWywr9IuQuncaAfUaS1g6sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@ampproject/remapping": "^2.2.0", + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.0", + "@babel/helper-compilation-targets": "^7.27.2", + "@babel/helper-module-transforms": "^7.27.3", + "@babel/helpers": "^7.27.6", + "@babel/parser": "^7.28.0", + "@babel/template": "^7.27.2", + "@babel/traverse": "^7.28.0", + "@babel/types": "^7.28.0", + "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/core/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/@babel/generator": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.0.tgz", + "integrity": "sha512-lJjzvrbEeWrhB4P3QBsH7tey117PjLZnDbLiQEKjQ/fNJTjuq4HSqgFA+UNSwZT8D7dxxbnuSBMsa1lrWzKlQg==", + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.28.0", + "@babel/types": "^7.28.0", + "@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==", + "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==", + "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.27.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.27.3.tgz", + "integrity": "sha512-dSOvYwvyLsWBeIRyOeHXp5vPj5l1I011r52FM1+r1jCERv+aFXYk4whgQccYEGYxK2H3ZAIA8nuPkQ0HaUo3qg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.27.1", + "@babel/helper-validator-identifier": "^7.27.1", + "@babel/traverse": "^7.27.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==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.27.1.tgz", + "integrity": "sha512-D2hP9eA+Sqx1kBZgzxZh0y1trbuU+JoDkiEwqhQ36nodYqJwyEIhPSdMNd7lOm/4io72luTPWH20Yda0xOuUow==", + "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.2", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.2.tgz", + "integrity": "sha512-/V9771t+EgXz62aCcyofnQhGM8DQACbRhvzKFsXKC9QM+5MadF8ZmIm0crDMaz3+o0h0zXfJnd4EhbYbxsrcFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.0.tgz", + "integrity": "sha512-jVZGvOxOuNSsuQuLRTh13nU0AogFlw32w/MT+LV6D3sP5WdbW61E77RnkbaO2dUvmPAYrBDJXGn5gGS6tH4j8g==", + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.0" + }, + "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/runtime": { + "version": "7.28.2", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.28.2.tgz", + "integrity": "sha512-KHp2IflsnGywDjBWDkR9iEqiWSpc8GIi0lgTT3mOElT0PP1tG26P4tmFI2YvAdzgq9RGyoHZQEIEdZy6Ec5xCA==", + "license": "MIT", + "engines": { + "node": ">=6.9.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==", + "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.0", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.0.tgz", + "integrity": "sha512-mGe7UK5wWyh0bKRfupsUchrQGqvDbZDbKJw+kcRGSmdHVYrv+ltd0pnpDTVpiTqnaBru9iEvA8pz8W46v0Amwg==", + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.0", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.28.0", + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.0", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.28.2", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.2.tgz", + "integrity": "sha512-ruv7Ae4J5dUYULmeXw1gmb7rYRz57OWCPM57pHojnLq/3Z1CK2lNSLTCVjxVk1F/TZHwOZZrOWi0ur95BbLxNQ==", + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@emotion/babel-plugin": { + "version": "11.13.5", + "resolved": "https://registry.npmjs.org/@emotion/babel-plugin/-/babel-plugin-11.13.5.tgz", + "integrity": "sha512-pxHCpT2ex+0q+HH91/zsdHkw/lXd468DIN2zvfvLtPKLLMo6gQj7oLObq8PhkrxOZb/gGCq03S3Z7PDhS8pduQ==", + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.16.7", + "@babel/runtime": "^7.18.3", + "@emotion/hash": "^0.9.2", + "@emotion/memoize": "^0.9.0", + "@emotion/serialize": "^1.3.3", + "babel-plugin-macros": "^3.1.0", + "convert-source-map": "^1.5.0", + "escape-string-regexp": "^4.0.0", + "find-root": "^1.1.0", + "source-map": "^0.5.7", + "stylis": "4.2.0" + } + }, + "node_modules/@emotion/cache": { + "version": "11.14.0", + "resolved": "https://registry.npmjs.org/@emotion/cache/-/cache-11.14.0.tgz", + "integrity": "sha512-L/B1lc/TViYk4DcpGxtAVbx0ZyiKM5ktoIyafGkH6zg/tj+mA+NE//aPYKG0k8kCHSHVJrpLpcAlOBEXQ3SavA==", + "license": "MIT", + "dependencies": { + "@emotion/memoize": "^0.9.0", + "@emotion/sheet": "^1.4.0", + "@emotion/utils": "^1.4.2", + "@emotion/weak-memoize": "^0.4.0", + "stylis": "4.2.0" + } + }, + "node_modules/@emotion/hash": { + "version": "0.9.2", + "resolved": "https://registry.npmjs.org/@emotion/hash/-/hash-0.9.2.tgz", + "integrity": "sha512-MyqliTZGuOm3+5ZRSaaBGP3USLw6+EGykkwZns2EPC5g8jJ4z9OrdZY9apkl3+UP9+sdz76YYkwCKP5gh8iY3g==", + "license": "MIT" + }, + "node_modules/@emotion/is-prop-valid": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/@emotion/is-prop-valid/-/is-prop-valid-1.3.1.tgz", + "integrity": "sha512-/ACwoqx7XQi9knQs/G0qKvv5teDMhD7bXYns9N/wM8ah8iNb8jZ2uNO0YOgiq2o2poIvVtJS2YALasQuMSQ7Kw==", + "license": "MIT", + "dependencies": { + "@emotion/memoize": "^0.9.0" + } + }, + "node_modules/@emotion/memoize": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/@emotion/memoize/-/memoize-0.9.0.tgz", + "integrity": "sha512-30FAj7/EoJ5mwVPOWhAyCX+FPfMDrVecJAM+Iw9NRoSl4BBAQeqj4cApHHUXOVvIPgLVDsCFoz/hGD+5QQD1GQ==", + "license": "MIT" + }, + "node_modules/@emotion/react": { + "version": "11.14.0", + "resolved": "https://registry.npmjs.org/@emotion/react/-/react-11.14.0.tgz", + "integrity": "sha512-O000MLDBDdk/EohJPFUqvnp4qnHeYkVP5B0xEG0D/L7cOKP9kefu2DXn8dj74cQfsEzUqh+sr1RzFqiL1o+PpA==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.18.3", + "@emotion/babel-plugin": "^11.13.5", + "@emotion/cache": "^11.14.0", + "@emotion/serialize": "^1.3.3", + "@emotion/use-insertion-effect-with-fallbacks": "^1.2.0", + "@emotion/utils": "^1.4.2", + "@emotion/weak-memoize": "^0.4.0", + "hoist-non-react-statics": "^3.3.1" + }, + "peerDependencies": { + "react": ">=16.8.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@emotion/serialize": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/@emotion/serialize/-/serialize-1.3.3.tgz", + "integrity": "sha512-EISGqt7sSNWHGI76hC7x1CksiXPahbxEOrC5RjmFRJTqLyEK9/9hZvBbiYn70dw4wuwMKiEMCUlR6ZXTSWQqxA==", + "license": "MIT", + "dependencies": { + "@emotion/hash": "^0.9.2", + "@emotion/memoize": "^0.9.0", + "@emotion/unitless": "^0.10.0", + "@emotion/utils": "^1.4.2", + "csstype": "^3.0.2" + } + }, + "node_modules/@emotion/sheet": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/@emotion/sheet/-/sheet-1.4.0.tgz", + "integrity": "sha512-fTBW9/8r2w3dXWYM4HCB1Rdp8NLibOw2+XELH5m5+AkWiL/KqYX6dc0kKYlaYyKjrQ6ds33MCdMPEwgs2z1rqg==", + "license": "MIT" + }, + "node_modules/@emotion/styled": { + "version": "11.14.1", + "resolved": "https://registry.npmjs.org/@emotion/styled/-/styled-11.14.1.tgz", + "integrity": "sha512-qEEJt42DuToa3gurlH4Qqc1kVpNq8wO8cJtDzU46TjlzWjDlsVyevtYCRijVq3SrHsROS+gVQ8Fnea108GnKzw==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.18.3", + "@emotion/babel-plugin": "^11.13.5", + "@emotion/is-prop-valid": "^1.3.0", + "@emotion/serialize": "^1.3.3", + "@emotion/use-insertion-effect-with-fallbacks": "^1.2.0", + "@emotion/utils": "^1.4.2" + }, + "peerDependencies": { + "@emotion/react": "^11.0.0-rc.0", + "react": ">=16.8.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@emotion/unitless": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/@emotion/unitless/-/unitless-0.10.0.tgz", + "integrity": "sha512-dFoMUuQA20zvtVTuxZww6OHoJYgrzfKM1t52mVySDJnMSEa08ruEvdYQbhvyu6soU+NeLVd3yKfTfT0NeV6qGg==", + "license": "MIT" + }, + "node_modules/@emotion/use-insertion-effect-with-fallbacks": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@emotion/use-insertion-effect-with-fallbacks/-/use-insertion-effect-with-fallbacks-1.2.0.tgz", + "integrity": "sha512-yJMtVdH59sxi/aVJBpk9FQq+OR8ll5GT8oWd57UpeaKEVGab41JWaCFA7FRLoMLloOZF/c/wsPoe+bfGmRKgDg==", + "license": "MIT", + "peerDependencies": { + "react": ">=16.8.0" + } + }, + "node_modules/@emotion/utils": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/@emotion/utils/-/utils-1.4.2.tgz", + "integrity": "sha512-3vLclRofFziIa3J2wDh9jjbkUz9qk5Vi3IZ/FSTKViB0k+ef0fPV7dYrUIugbgupYDx7v9ud/SjrtEP8Y4xLoA==", + "license": "MIT" + }, + "node_modules/@emotion/weak-memoize": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/@emotion/weak-memoize/-/weak-memoize-0.4.0.tgz", + "integrity": "sha512-snKqtPW01tN0ui7yu9rGv69aJXr/a/Ywvl11sUjNtEcRc+ng/mQriFL0wLXMef74iHa/EkftbDzU9F8iFbH+zg==", + "license": "MIT" + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.9.tgz", + "integrity": "sha512-OaGtL73Jck6pBKjNIe24BnFE6agGl+6KxDtTfHhy1HmhthfKouEcOhqpSL64K4/0WCtbKFLOdzD/44cJ4k9opA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.9.tgz", + "integrity": "sha512-5WNI1DaMtxQ7t7B6xa572XMXpHAaI/9Hnhk8lcxF4zVN4xstUgTlvuGDorBguKEnZO70qwEcLpfifMLoxiPqHQ==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.9.tgz", + "integrity": "sha512-IDrddSmpSv51ftWslJMvl3Q2ZT98fUSL2/rlUXuVqRXHCs5EUF1/f+jbjF5+NG9UffUDMCiTyh8iec7u8RlTLg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.9.tgz", + "integrity": "sha512-I853iMZ1hWZdNllhVZKm34f4wErd4lMyeV7BLzEExGEIZYsOzqDWDf+y082izYUE8gtJnYHdeDpN/6tUdwvfiw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.9.tgz", + "integrity": "sha512-XIpIDMAjOELi/9PB30vEbVMs3GV1v2zkkPnuyRRURbhqjyzIINwj+nbQATh4H9GxUgH1kFsEyQMxwiLFKUS6Rg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.9.tgz", + "integrity": "sha512-jhHfBzjYTA1IQu8VyrjCX4ApJDnH+ez+IYVEoJHeqJm9VhG9Dh2BYaJritkYK3vMaXrf7Ogr/0MQ8/MeIefsPQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.9.tgz", + "integrity": "sha512-z93DmbnY6fX9+KdD4Ue/H6sYs+bhFQJNCPZsi4XWJoYblUqT06MQUdBCpcSfuiN72AbqeBFu5LVQTjfXDE2A6Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.9.tgz", + "integrity": "sha512-mrKX6H/vOyo5v71YfXWJxLVxgy1kyt1MQaD8wZJgJfG4gq4DpQGpgTB74e5yBeQdyMTbgxp0YtNj7NuHN0PoZg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.9.tgz", + "integrity": "sha512-HBU2Xv78SMgaydBmdor38lg8YDnFKSARg1Q6AT0/y2ezUAKiZvc211RDFHlEZRFNRVhcMamiToo7bDx3VEOYQw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.9.tgz", + "integrity": "sha512-BlB7bIcLT3G26urh5Dmse7fiLmLXnRlopw4s8DalgZ8ef79Jj4aUcYbk90g8iCa2467HX8SAIidbL7gsqXHdRw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.9.tgz", + "integrity": "sha512-e7S3MOJPZGp2QW6AK6+Ly81rC7oOSerQ+P8L0ta4FhVi+/j/v2yZzx5CqqDaWjtPFfYz21Vi1S0auHrap3Ma3A==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.9.tgz", + "integrity": "sha512-Sbe10Bnn0oUAB2AalYztvGcK+o6YFFA/9829PhOCUS9vkJElXGdphz0A3DbMdP8gmKkqPmPcMJmJOrI3VYB1JQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.9.tgz", + "integrity": "sha512-YcM5br0mVyZw2jcQeLIkhWtKPeVfAerES5PvOzaDxVtIyZ2NUBZKNLjC5z3/fUlDgT6w89VsxP2qzNipOaaDyA==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.9.tgz", + "integrity": "sha512-++0HQvasdo20JytyDpFvQtNrEsAgNG2CY1CLMwGXfFTKGBGQT3bOeLSYE2l1fYdvML5KUuwn9Z8L1EWe2tzs1w==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.9.tgz", + "integrity": "sha512-uNIBa279Y3fkjV+2cUjx36xkx7eSjb8IvnL01eXUKXez/CBHNRw5ekCGMPM0BcmqBxBcdgUWuUXmVWwm4CH9kg==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.9.tgz", + "integrity": "sha512-Mfiphvp3MjC/lctb+7D287Xw1DGzqJPb/J2aHHcHxflUo+8tmN/6d4k6I2yFR7BVo5/g7x2Monq4+Yew0EHRIA==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.9.tgz", + "integrity": "sha512-iSwByxzRe48YVkmpbgoxVzn76BXjlYFXC7NvLYq+b+kDjyyk30J0JY47DIn8z1MO3K0oSl9fZoRmZPQI4Hklzg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.9.tgz", + "integrity": "sha512-9jNJl6FqaUG+COdQMjSCGW4QiMHH88xWbvZ+kRVblZsWrkXlABuGdFJ1E9L7HK+T0Yqd4akKNa/lO0+jDxQD4Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.9.tgz", + "integrity": "sha512-RLLdkflmqRG8KanPGOU7Rpg829ZHu8nFy5Pqdi9U01VYtG9Y0zOG6Vr2z4/S+/3zIyOxiK6cCeYNWOFR9QP87g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.9.tgz", + "integrity": "sha512-YaFBlPGeDasft5IIM+CQAhJAqS3St3nJzDEgsgFixcfZeyGPCd6eJBWzke5piZuZ7CtL656eOSYKk4Ls2C0FRQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.9.tgz", + "integrity": "sha512-1MkgTCuvMGWuqVtAvkpkXFmtL8XhWy+j4jaSO2wxfJtilVCi0ZE37b8uOdMItIHz4I6z1bWWtEX4CJwcKYLcuA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openharmony-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.9.tgz", + "integrity": "sha512-4Xd0xNiMVXKh6Fa7HEJQbrpP3m3DDn43jKxMjxLLRjWnRsfxjORYJlXPO4JNcXtOyfajXorRKY9NkOpTHptErg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.9.tgz", + "integrity": "sha512-WjH4s6hzo00nNezhp3wFIAfmGZ8U7KtrJNlFMRKxiI9mxEK1scOMAaa9i4crUtu+tBr+0IN6JCuAcSBJZfnphw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.9.tgz", + "integrity": "sha512-mGFrVJHmZiRqmP8xFOc6b84/7xa5y5YvR1x8djzXpJBSv/UsNK6aqec+6JDjConTgvvQefdGhFDAs2DLAds6gQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.9.tgz", + "integrity": "sha512-b33gLVU2k11nVx1OhX3C8QQP6UHQK4ZtN56oFWvVXvz2VkDoe6fbG8TOgHFxEvqeqohmRnIHe5A1+HADk4OQww==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.9.tgz", + "integrity": "sha512-PPOl1mi6lpLNQxnGoyAfschAodRFYXJ+9fs6WHXz7CSWKbOqiMZsubC+BQsVKuul+3vKLuwTHsS2c2y9EoKwxQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.7.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.7.0.tgz", + "integrity": "sha512-dyybb3AcajC7uha6CvhdVRJqaKyn7w2YKqKyAN37NKYgZT36w+iRb0Dymmc5qEJ549c/S31cMMSFd75bteCpCw==", + "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.1", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", + "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/config-array": { + "version": "0.21.0", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.0.tgz", + "integrity": "sha512-ENIdc4iLu0d93HeYirvKmrzshzofPw6VkZRKQGe9Nv46ZnWUzcF1xV01dcvEg/1wXUR61OmmlSfyeyO7EvjLxQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/object-schema": "^2.1.6", + "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.3.1", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.3.1.tgz", + "integrity": "sha512-xR93k9WhrDYpXHORXpxVL5oHj3Era7wo6k/Wd8/IsQNnZUTzkGS29lyn3nAT05v6ltUuTFVCCYDEGfy2Or/sPA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/core": { + "version": "0.15.2", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.15.2.tgz", + "integrity": "sha512-78Md3/Rrxh83gCxoUc0EiciuOHsIITzLy53m3d9UyiW8y9Dj2D29FeETqyKA+BRK76tnTp6RXWb3pCay8Oyomg==", + "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.33.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.33.0.tgz", + "integrity": "sha512-5K1/mKhWaMfreBGJTwval43JJmkip0RmM+3+IuqupeSKNC/Th2Kc7ucaq5ovTSra/OOKB9c58CGSz3QMVbWt0A==", + "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.6", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.6.tgz", + "integrity": "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/plugin-kit": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.3.5.tgz", + "integrity": "sha512-Z5kJ+wU3oA7MMIqVR9tyZRtjYPr4OC004Q4Rw7pgOKUOKkJfZ3O24nz3WYfGRpMDNmcOi3TwQOmgm7B7Tpii0w==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.15.2", + "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.6", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.6.tgz", + "integrity": "sha512-YuI2ZHQL78Q5HbhDiBA1X4LmYdXCKCMQIfw0pw7piHJwyREFebJUvrQN4cMssyES6x+vfUbx1CIpaQUKYdQZOw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanfs/core": "^0.19.1", + "@humanwhocodes/retry": "^0.3.0" + }, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node/node_modules/@humanwhocodes/retry": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz", + "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "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==", + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@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==", + "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==", + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.30", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.30.tgz", + "integrity": "sha512-GQ7Nw5G2lTu/BtHTKfXhKHok2WGetd4XYcVKGx00SjAk8GMwgJM3zr6zORiPGuOE+/vkc90KtTosSSvaCjKb2Q==", + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@mui/core-downloads-tracker": { + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/@mui/core-downloads-tracker/-/core-downloads-tracker-7.3.1.tgz", + "integrity": "sha512-+mIK1Z0BhOaQ0vCgOkT1mSrIpEHLo338h4/duuL4TBLXPvUMit732mnwJY3W40Avy30HdeSfwUAAGRkKmwRaEQ==", + "license": "MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + } + }, + "node_modules/@mui/icons-material": { + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/@mui/icons-material/-/icons-material-7.3.1.tgz", + "integrity": "sha512-upzCtG6awpL6noEZlJ5Z01khZ9VnLNLaj7tb6iPbN6G97eYfUTs8e9OyPKy3rEms3VQWmVBfri7jzeaRxdFIzA==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.28.2" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@mui/material": "^7.3.1", + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/material": { + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/@mui/material/-/material-7.3.1.tgz", + "integrity": "sha512-Xf6Shbo03YmcBedZMwSpEFOwpYDtU7tC+rhAHTrA9FHk0FpsDqiQ9jUa1j/9s3HLs7KWb5mDcGnlwdh9Q9KAag==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.28.2", + "@mui/core-downloads-tracker": "^7.3.1", + "@mui/system": "^7.3.1", + "@mui/types": "^7.4.5", + "@mui/utils": "^7.3.1", + "@popperjs/core": "^2.11.8", + "@types/react-transition-group": "^4.4.12", + "clsx": "^2.1.1", + "csstype": "^3.1.3", + "prop-types": "^15.8.1", + "react-is": "^19.1.1", + "react-transition-group": "^4.4.5" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@emotion/react": "^11.5.0", + "@emotion/styled": "^11.3.0", + "@mui/material-pigment-css": "^7.3.1", + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react-dom": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/react": { + "optional": true + }, + "@emotion/styled": { + "optional": true + }, + "@mui/material-pigment-css": { + "optional": true + }, + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/private-theming": { + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/@mui/private-theming/-/private-theming-7.3.1.tgz", + "integrity": "sha512-WU3YLkKXii/x8ZEKnrLKsPwplCVE11yZxUvlaaZSIzCcI3x2OdFC8eMlNy74hVeUsYQvzzX1Es/k4ARPlFvpPQ==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.28.2", + "@mui/utils": "^7.3.1", + "prop-types": "^15.8.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/styled-engine": { + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/@mui/styled-engine/-/styled-engine-7.3.1.tgz", + "integrity": "sha512-Nqo6OHjvJpXJ1+9TekTE//+8RybgPQUKwns2Lh0sq+8rJOUSUKS3KALv4InSOdHhIM9Mdi8/L7LTF1/Ky6D6TQ==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.28.2", + "@emotion/cache": "^11.14.0", + "@emotion/serialize": "^1.3.3", + "@emotion/sheet": "^1.4.0", + "csstype": "^3.1.3", + "prop-types": "^15.8.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@emotion/react": "^11.4.1", + "@emotion/styled": "^11.3.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/react": { + "optional": true + }, + "@emotion/styled": { + "optional": true + } + } + }, + "node_modules/@mui/system": { + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/@mui/system/-/system-7.3.1.tgz", + "integrity": "sha512-mIidecvcNVpNJMdPDmCeoSL5zshKBbYPcphjuh6ZMjhybhqhZ4mX6k9zmIWh6XOXcqRQMg5KrcjnO0QstrNj3w==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.28.2", + "@mui/private-theming": "^7.3.1", + "@mui/styled-engine": "^7.3.1", + "@mui/types": "^7.4.5", + "@mui/utils": "^7.3.1", + "clsx": "^2.1.1", + "csstype": "^3.1.3", + "prop-types": "^15.8.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@emotion/react": "^11.5.0", + "@emotion/styled": "^11.3.0", + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/react": { + "optional": true + }, + "@emotion/styled": { + "optional": true + }, + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/types": { + "version": "7.4.5", + "resolved": "https://registry.npmjs.org/@mui/types/-/types-7.4.5.tgz", + "integrity": "sha512-ZPwlAOE3e8C0piCKbaabwrqZbW4QvWz0uapVPWya7fYj6PeDkl5sSJmomT7wjOcZGPB48G/a6Ubidqreptxz4g==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.28.2" + }, + "peerDependencies": { + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/utils": { + "version": "7.3.1", + "resolved": "https://registry.npmjs.org/@mui/utils/-/utils-7.3.1.tgz", + "integrity": "sha512-/31y4wZqVWa0jzMnzo6JPjxwP6xXy4P3+iLbosFg/mJQowL1KIou0LC+lquWW60FKVbKz5ZUWBg2H3jausa0pw==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.28.2", + "@mui/types": "^7.4.5", + "@types/prop-types": "^15.7.15", + "clsx": "^2.1.1", + "prop-types": "^15.8.1", + "react-is": "^19.1.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@popperjs/core": { + "version": "2.11.8", + "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.11.8.tgz", + "integrity": "sha512-P1st0aksCrn9sGZhp8GMYwBnQsbvAWsZAX44oXNNvLHGqAOcoVxmjZiohstwQ7SqKnbR47akdNi+uleWD8+g6A==", + "license": "MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/popperjs" + } + }, + "node_modules/@rolldown/pluginutils": { + "version": "1.0.0-beta.30", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.30.tgz", + "integrity": "sha512-whXaSoNUFiyDAjkUF8OBpOm77Szdbk5lGNqFe6CbVbJFrhCCPinCbRA3NjawwlNHla1No7xvXXh+CpSxnPfUEw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.46.2.tgz", + "integrity": "sha512-Zj3Hl6sN34xJtMv7Anwb5Gu01yujyE/cLBDB2gnHTAHaWS1Z38L7kuSG+oAh0giZMqG060f/YBStXtMH6FvPMA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.46.2.tgz", + "integrity": "sha512-nTeCWY83kN64oQ5MGz3CgtPx8NSOhC5lWtsjTs+8JAJNLcP3QbLCtDDgUKQc/Ro/frpMq4SHUaHN6AMltcEoLQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.46.2.tgz", + "integrity": "sha512-HV7bW2Fb/F5KPdM/9bApunQh68YVDU8sO8BvcW9OngQVN3HHHkw99wFupuUJfGR9pYLLAjcAOA6iO+evsbBaPQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.46.2.tgz", + "integrity": "sha512-SSj8TlYV5nJixSsm/y3QXfhspSiLYP11zpfwp6G/YDXctf3Xkdnk4woJIF5VQe0of2OjzTt8EsxnJDCdHd2xMA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.46.2.tgz", + "integrity": "sha512-ZyrsG4TIT9xnOlLsSSi9w/X29tCbK1yegE49RYm3tu3wF1L/B6LVMqnEWyDB26d9Ecx9zrmXCiPmIabVuLmNSg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.46.2.tgz", + "integrity": "sha512-pCgHFoOECwVCJ5GFq8+gR8SBKnMO+xe5UEqbemxBpCKYQddRQMgomv1104RnLSg7nNvgKy05sLsY51+OVRyiVw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.46.2.tgz", + "integrity": "sha512-EtP8aquZ0xQg0ETFcxUbU71MZlHaw9MChwrQzatiE8U/bvi5uv/oChExXC4mWhjiqK7azGJBqU0tt5H123SzVA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.46.2.tgz", + "integrity": "sha512-qO7F7U3u1nfxYRPM8HqFtLd+raev2K137dsV08q/LRKRLEc7RsiDWihUnrINdsWQxPR9jqZ8DIIZ1zJJAm5PjQ==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.46.2.tgz", + "integrity": "sha512-3dRaqLfcOXYsfvw5xMrxAk9Lb1f395gkoBYzSFcc/scgRFptRXL9DOaDpMiehf9CO8ZDRJW2z45b6fpU5nwjng==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.46.2.tgz", + "integrity": "sha512-fhHFTutA7SM+IrR6lIfiHskxmpmPTJUXpWIsBXpeEwNgZzZZSg/q4i6FU4J8qOGyJ0TR+wXBwx/L7Ho9z0+uDg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loongarch64-gnu": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.46.2.tgz", + "integrity": "sha512-i7wfGFXu8x4+FRqPymzjD+Hyav8l95UIZ773j7J7zRYc3Xsxy2wIn4x+llpunexXe6laaO72iEjeeGyUFmjKeA==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-gnu": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.46.2.tgz", + "integrity": "sha512-B/l0dFcHVUnqcGZWKcWBSV2PF01YUt0Rvlurci5P+neqY/yMKchGU8ullZvIv5e8Y1C6wOn+U03mrDylP5q9Yw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.46.2.tgz", + "integrity": "sha512-32k4ENb5ygtkMwPMucAb8MtV8olkPT03oiTxJbgkJa7lJ7dZMr0GCFJlyvy+K8iq7F/iuOr41ZdUHaOiqyR3iQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.46.2.tgz", + "integrity": "sha512-t5B2loThlFEauloaQkZg9gxV05BYeITLvLkWOkRXogP4qHXLkWSbSHKM9S6H1schf/0YGP/qNKtiISlxvfmmZw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.46.2.tgz", + "integrity": "sha512-YKjekwTEKgbB7n17gmODSmJVUIvj8CX7q5442/CK80L8nqOUbMtf8b01QkG3jOqyr1rotrAnW6B/qiHwfcuWQA==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.46.2.tgz", + "integrity": "sha512-Jj5a9RUoe5ra+MEyERkDKLwTXVu6s3aACP51nkfnK9wJTraCC8IMe3snOfALkrjTYd2G1ViE1hICj0fZ7ALBPA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.46.2.tgz", + "integrity": "sha512-7kX69DIrBeD7yNp4A5b81izs8BqoZkCIaxQaOpumcJ1S/kmqNFjPhDu1LHeVXv0SexfHQv5cqHsxLOjETuqDuA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.46.2.tgz", + "integrity": "sha512-wiJWMIpeaak/jsbaq2HMh/rzZxHVW1rU6coyeNNpMwk5isiPjSTx0a4YLSlYDwBH/WBvLz+EtsNqQScZTLJy3g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.46.2.tgz", + "integrity": "sha512-gBgaUDESVzMgWZhcyjfs9QFK16D8K6QZpwAaVNJxYDLHWayOta4ZMjGm/vsAEy3hvlS2GosVFlBlP9/Wb85DqQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.46.2.tgz", + "integrity": "sha512-CvUo2ixeIQGtF6WvuB87XWqPQkoFAFqW+HUo/WzHwuHDvIwZCtjdWXoYCcr06iKGydiqTclC4jU/TNObC/xKZg==", + "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/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/parse-json": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.2.tgz", + "integrity": "sha512-dISoDXWWQwUquiKsyZ4Ng+HX2KsPL7LyHKHQwgGFEA3IaKac4Obd+h2a/a6waisAoepJlBcx9paWqjA8/HVjCw==", + "license": "MIT" + }, + "node_modules/@types/prop-types": { + "version": "15.7.15", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz", + "integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==", + "license": "MIT" + }, + "node_modules/@types/react": { + "version": "19.1.10", + "resolved": "https://registry.npmjs.org/@types/react/-/react-19.1.10.tgz", + "integrity": "sha512-EhBeSYX0Y6ye8pNebpKrwFJq7BoQ8J5SO6NlvNwwHjSj6adXJViPQrKlsyPw7hLBLvckEMO1yxeGdR82YBBlDg==", + "license": "MIT", + "dependencies": { + "csstype": "^3.0.2" + } + }, + "node_modules/@types/react-dom": { + "version": "19.1.7", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.1.7.tgz", + "integrity": "sha512-i5ZzwYpqjmrKenzkoLM2Ibzt6mAsM7pxB6BCIouEVVmgiqaMj1TjaK7hnA36hbW5aZv20kx7Lw6hWzPWg0Rurw==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "@types/react": "^19.0.0" + } + }, + "node_modules/@types/react-transition-group": { + "version": "4.4.12", + "resolved": "https://registry.npmjs.org/@types/react-transition-group/-/react-transition-group-4.4.12.tgz", + "integrity": "sha512-8TV6R3h2j7a91c+1DXdJi3Syo69zzIZbz7Lg5tORM5LEJG7X/E6a1V3drRyBRZq7/utz7A+c4OgYLiLcYGHG6w==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*" + } + }, + "node_modules/@vitejs/plugin-react": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-5.0.0.tgz", + "integrity": "sha512-Jx9JfsTa05bYkS9xo0hkofp2dCmp1blrKjw9JONs5BTHOvJCgLbaPSuZLGSVJW6u2qe0tc4eevY0+gSNNi0YCw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.28.0", + "@babel/plugin-transform-react-jsx-self": "^7.27.1", + "@babel/plugin-transform-react-jsx-source": "^7.27.1", + "@rolldown/pluginutils": "1.0.0-beta.30", + "@types/babel__core": "^7.20.5", + "react-refresh": "^0.17.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.11.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.11.0.tgz", + "integrity": "sha512-1Lx3WLFQWm3ooKDYZD1eXmoGO9fxYQjrycfHFC8P0sCfQVXyROp0p9PFWBehewBOdCwHc+f/b8I0fMto5eSfwA==", + "license": "MIT", + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.4", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/babel-plugin-macros": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/babel-plugin-macros/-/babel-plugin-macros-3.1.0.tgz", + "integrity": "sha512-Cg7TFGpIr01vOQNODXOOaGz2NpCU5gl8x1qJFbb6hbZxR7XrcE2vtbAsTAbJ7/xwJtUuJEw8K8Zr/AE0LHlesg==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.12.5", + "cosmiconfig": "^7.0.0", + "resolve": "^1.19.0" + }, + "engines": { + "node": ">=10", + "npm": ">=6" + } + }, + "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/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/browserslist": { + "version": "4.25.2", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.25.2.tgz", + "integrity": "sha512-0si2SJK3ooGzIawRu61ZdPCO1IncZwS8IzuX73sPZsXW6EQ/w/DAfPyKI8l1ETTCr2MnvqWitmlCUxgdul45jA==", + "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": { + "caniuse-lite": "^1.0.30001733", + "electron-to-chromium": "^1.5.199", + "node-releases": "^2.0.19", + "update-browserslist-db": "^1.1.3" + }, + "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==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001734", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001734.tgz", + "integrity": "sha512-uhE1Ye5vgqju6OI71HTQqcBCZrvHugk0MjLak7Q+HfoBgoq5Bi+5YnwjP4fjDgrtYr/l8MVRBvzz9dPD4KyK0A==", + "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/clsx": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", + "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "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": "1.9.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.9.0.tgz", + "integrity": "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==", + "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/cosmiconfig": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.1.0.tgz", + "integrity": "sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==", + "license": "MIT", + "dependencies": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.2.1", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.10.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/cosmiconfig/node_modules/yaml": { + "version": "1.10.2", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", + "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", + "license": "ISC", + "engines": { + "node": ">= 6" + } + }, + "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.1.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", + "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", + "license": "MIT" + }, + "node_modules/debug": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.1.tgz", + "integrity": "sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==", + "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/dom-helpers": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/dom-helpers/-/dom-helpers-5.2.1.tgz", + "integrity": "sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.8.7", + "csstype": "^3.0.2" + } + }, + "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.200", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.200.tgz", + "integrity": "sha512-rFCxROw7aOe4uPTfIAx+rXv9cEcGx+buAF4npnhtTqCJk5KDFRnh3+KYj7rdVh6lsFt5/aPs+Irj9rZ33WMA7w==", + "dev": true, + "license": "ISC" + }, + "node_modules/error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "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.9", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.9.tgz", + "integrity": "sha512-CRbODhYyQx3qp7ZEwzxOk4JBqmD/seJrzPa/cGjY1VtIn5E09Oi9/dB4JwctnfZ8Q8iT7rioVv5k/FNT/uf54g==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.25.9", + "@esbuild/android-arm": "0.25.9", + "@esbuild/android-arm64": "0.25.9", + "@esbuild/android-x64": "0.25.9", + "@esbuild/darwin-arm64": "0.25.9", + "@esbuild/darwin-x64": "0.25.9", + "@esbuild/freebsd-arm64": "0.25.9", + "@esbuild/freebsd-x64": "0.25.9", + "@esbuild/linux-arm": "0.25.9", + "@esbuild/linux-arm64": "0.25.9", + "@esbuild/linux-ia32": "0.25.9", + "@esbuild/linux-loong64": "0.25.9", + "@esbuild/linux-mips64el": "0.25.9", + "@esbuild/linux-ppc64": "0.25.9", + "@esbuild/linux-riscv64": "0.25.9", + "@esbuild/linux-s390x": "0.25.9", + "@esbuild/linux-x64": "0.25.9", + "@esbuild/netbsd-arm64": "0.25.9", + "@esbuild/netbsd-x64": "0.25.9", + "@esbuild/openbsd-arm64": "0.25.9", + "@esbuild/openbsd-x64": "0.25.9", + "@esbuild/openharmony-arm64": "0.25.9", + "@esbuild/sunos-x64": "0.25.9", + "@esbuild/win32-arm64": "0.25.9", + "@esbuild/win32-ia32": "0.25.9", + "@esbuild/win32-x64": "0.25.9" + } + }, + "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==", + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "9.33.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.33.0.tgz", + "integrity": "sha512-TS9bTNIryDzStCpJN93aC5VRSW3uTx9sClUn4B87pwiCaJh220otoI0X8mJKr+VcPtniMdN8GKjlwgWGUv5ZKA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.12.1", + "@eslint/config-array": "^0.21.0", + "@eslint/config-helpers": "^0.3.1", + "@eslint/core": "^0.15.2", + "@eslint/eslintrc": "^3.3.1", + "@eslint/js": "9.33.0", + "@eslint/plugin-kit": "^0.3.5", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.2", + "@types/estree": "^1.0.6", + "@types/json-schema": "^7.0.15", + "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": "5.2.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-5.2.0.tgz", + "integrity": "sha512-+f15FfK64YQwZdJNELETdn5ibXEUQmW1DZL6KXhNnc2heoy/sg9VJJeT7n8TlMWouzWqSWavFkIhHyIbIAEapg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "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.20", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.20.tgz", + "integrity": "sha512-XpbHQ2q5gUF8BGOX4dHe+71qoirYMhApEPZ7sfhF/dNnOF1UXnCMGZf79SFTBO7Bz5YEIT4TMieSlJBWhP9WBA==", + "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-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/fdir": { + "version": "6.4.6", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.6.tgz", + "integrity": "sha512-hiFoqpyZcfNm1yc4u8oWCf9A2c4D3QjCrks3zmoVKVxpQRzmPNar1hUJcBG2RQHvEVGDN+Jm81ZheVLAQMK6+w==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "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/find-root": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/find-root/-/find-root-1.1.0.tgz", + "integrity": "sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng==", + "license": "MIT" + }, + "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.4", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.4.tgz", + "integrity": "sha512-KrGhL9Q4zjj0kiUt5OO4Mr/A/jlI2jDYs5eHBpYHPcBEVSiipAvn2Ko2HnPe20rmcuuvMHNdZFp+4IlGTMF0Ow==", + "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.3.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-16.3.0.tgz", + "integrity": "sha512-bqWEnJ1Nt3neqx2q5SFfGS8r/ahumIakg3HcwtNlrVlwXIeNumWn/c7Pn/wKzGhf6SaW6H6uWXLqC30STCMchQ==", + "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/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/hoist-non-react-statics": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz", + "integrity": "sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==", + "license": "BSD-3-Clause", + "dependencies": { + "react-is": "^16.7.0" + } + }, + "node_modules/hoist-non-react-statics/node_modules/react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", + "license": "MIT" + }, + "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==", + "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-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "license": "MIT" + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/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==", + "license": "MIT" + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "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==", + "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-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "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/jwt-decode": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-4.0.0.tgz", + "integrity": "sha512-+KJGIyHgkGuIq3IEBNftfhW/LfWhXUIY6OmyVWjliu5KH1y0fw7VQ8YndE2O4qZdMSd9SqbnC8GOcZEy0Om7sA==", + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "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/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "license": "MIT" + }, + "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/loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "license": "MIT", + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "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/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/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==", + "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.19", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.19.tgz", + "integrity": "sha512-xxOWJsBKtzAq7DY0J+DTzuz58K8e7sJbdgwkbMWQe8UYB6ekmsQ45q0M/tJDsGaZmbC+l7n57UV8Hl5tHxO9uw==", + "dev": true, + "license": "MIT" + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "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==", + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "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/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "license": "MIT" + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "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==", + "license": "ISC" + }, + "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/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/prop-types": { + "version": "15.8.1", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz", + "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==", + "license": "MIT", + "dependencies": { + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.13.1" + } + }, + "node_modules/prop-types/node_modules/react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", + "license": "MIT" + }, + "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/react": { + "version": "19.1.1", + "resolved": "https://registry.npmjs.org/react/-/react-19.1.1.tgz", + "integrity": "sha512-w8nqGImo45dmMIfljjMwOGtbmC/mk4CMYhWIicdSflH91J9TyCyczcPFXJzrZ/ZXcgGRFeP6BU0BEJTw6tZdfQ==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "19.1.1", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.1.1.tgz", + "integrity": "sha512-Dlq/5LAZgF0Gaz6yiqZCf6VCcZs1ghAJyrsu84Q/GT0gV+mCxbfmKNoGRKBYMJ8IEdGPqu49YWXD02GCknEDkw==", + "license": "MIT", + "dependencies": { + "scheduler": "^0.26.0" + }, + "peerDependencies": { + "react": "^19.1.1" + } + }, + "node_modules/react-is": { + "version": "19.1.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-19.1.1.tgz", + "integrity": "sha512-tr41fA15Vn8p4X9ntI+yCyeGSf1TlYaY5vlTZfQmeLBrFo3psOPX6HhTDnFNL9uj3EhP0KAQ80cugCl4b4BERA==", + "license": "MIT" + }, + "node_modules/react-refresh": { + "version": "0.17.0", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.17.0.tgz", + "integrity": "sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-router": { + "version": "7.8.0", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-7.8.0.tgz", + "integrity": "sha512-r15M3+LHKgM4SOapNmsH3smAizWds1vJ0Z9C4mWaKnT9/wD7+d/0jYcj6LmOvonkrO4Rgdyp4KQ/29gWN2i1eg==", + "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.8.0", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-7.8.0.tgz", + "integrity": "sha512-ntInsnDVnVRdtSu6ODmTQ41cbluak/ENeTif7GBce0L6eztFg6/e1hXAysFQI8X25C8ipKmT9cClbJwxx3Kaqw==", + "license": "MIT", + "dependencies": { + "react-router": "7.8.0" + }, + "engines": { + "node": ">=20.0.0" + }, + "peerDependencies": { + "react": ">=18", + "react-dom": ">=18" + } + }, + "node_modules/react-transition-group": { + "version": "4.4.5", + "resolved": "https://registry.npmjs.org/react-transition-group/-/react-transition-group-4.4.5.tgz", + "integrity": "sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g==", + "license": "BSD-3-Clause", + "dependencies": { + "@babel/runtime": "^7.5.5", + "dom-helpers": "^5.0.1", + "loose-envify": "^1.4.0", + "prop-types": "^15.6.2" + }, + "peerDependencies": { + "react": ">=16.6.0", + "react-dom": ">=16.6.0" + } + }, + "node_modules/resolve": { + "version": "1.22.10", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", + "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", + "license": "MIT", + "dependencies": { + "is-core-module": "^2.16.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/rollup": { + "version": "4.46.2", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.46.2.tgz", + "integrity": "sha512-WMmLFI+Boh6xbop+OAGo9cQ3OgX9MIg7xOQjn+pTCwOkk+FNDAeAemXkJ3HzDJrVXleLOFVa1ipuc1AmEx1Dwg==", + "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.46.2", + "@rollup/rollup-android-arm64": "4.46.2", + "@rollup/rollup-darwin-arm64": "4.46.2", + "@rollup/rollup-darwin-x64": "4.46.2", + "@rollup/rollup-freebsd-arm64": "4.46.2", + "@rollup/rollup-freebsd-x64": "4.46.2", + "@rollup/rollup-linux-arm-gnueabihf": "4.46.2", + "@rollup/rollup-linux-arm-musleabihf": "4.46.2", + "@rollup/rollup-linux-arm64-gnu": "4.46.2", + "@rollup/rollup-linux-arm64-musl": "4.46.2", + "@rollup/rollup-linux-loongarch64-gnu": "4.46.2", + "@rollup/rollup-linux-ppc64-gnu": "4.46.2", + "@rollup/rollup-linux-riscv64-gnu": "4.46.2", + "@rollup/rollup-linux-riscv64-musl": "4.46.2", + "@rollup/rollup-linux-s390x-gnu": "4.46.2", + "@rollup/rollup-linux-x64-gnu": "4.46.2", + "@rollup/rollup-linux-x64-musl": "4.46.2", + "@rollup/rollup-win32-arm64-msvc": "4.46.2", + "@rollup/rollup-win32-ia32-msvc": "4.46.2", + "@rollup/rollup-win32-x64-msvc": "4.46.2", + "fsevents": "~2.3.2" + } + }, + "node_modules/scheduler": { + "version": "0.26.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.26.0.tgz", + "integrity": "sha512-NlHwttCI/l5gCPR3D1nNXtWABUmBwvZpEQiD4IXSbIDq8BzLIK/7Ir5gTFSGZDUu37K5cMNp0hFtzO38sC7gWA==", + "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.1", + "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.1.tgz", + "integrity": "sha512-IOc8uWeOZgnb3ptbCURJWNjWUPcO3ZnTTdzsurqERrP6nPyv+paC55vJM0LpOlT2ne+Ix+9+CRG1MNLlyZ4GjQ==", + "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": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==", + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "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/stylis": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/stylis/-/stylis-4.2.0.tgz", + "integrity": "sha512-Orov6g6BB1sDfYgzWfTHDOxamtX1bE/zo104Dh9e6fqJ3PooipYyfJ0pUmrZO2wAvO8YbEyeFrkV91XTsGMSrw==", + "license": "MIT" + }, + "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/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/tinyglobby": { + "version": "0.2.14", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.14.tgz", + "integrity": "sha512-tX5e7OM1HnYr2+a2C/4V0htOcSQcoSTH9KgJnVvNm5zm/cyEWKJ7j7YutsH9CxMdtOkkLFy2AHrMci9IM8IPZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.4.4", + "picomatch": "^4.0.2" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, + "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/update-browserslist-db": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.3.tgz", + "integrity": "sha512-UxhIZQ+QInVdunkDAaiazvvT/+fXL5Osr0JZlJulepYu6Jd7qJtDZjlur0emRlT71EN3ScPoE7gvsuIKKNavKw==", + "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.1.2", + "resolved": "https://registry.npmjs.org/vite/-/vite-7.1.2.tgz", + "integrity": "sha512-J0SQBPlQiEXAF7tajiH+rUooJPo0l8KQgyg4/aMunNtrOa7bwuZJsJbDWzeljqQpgftxuq5yNJxQ91O9ts29UQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "^0.25.0", + "fdir": "^6.4.6", + "picomatch": "^4.0.3", + "postcss": "^8.5.6", + "rollup": "^4.43.0", + "tinyglobby": "^0.2.14" + }, + "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/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" + } + } + } +} diff --git a/frontend/package.json b/frontend/package.json new file mode 100644 index 0000000..c5ca0fd --- /dev/null +++ b/frontend/package.json @@ -0,0 +1,34 @@ +{ + "name": "frontend", + "private": true, + "version": "0.0.0", + "type": "module", + "scripts": { + "dev": "vite", + "build": "vite build", + "lint": "eslint .", + "preview": "vite preview" + }, + "dependencies": { + "@emotion/react": "^11.14.0", + "@emotion/styled": "^11.14.1", + "@mui/icons-material": "^7.3.1", + "@mui/material": "^7.3.1", + "axios": "^1.11.0", + "jwt-decode": "^4.0.0", + "react": "^19.1.1", + "react-dom": "^19.1.1", + "react-router-dom": "^7.8.0" + }, + "devDependencies": { + "@eslint/js": "^9.33.0", + "@types/react": "^19.1.10", + "@types/react-dom": "^19.1.7", + "@vitejs/plugin-react": "^5.0.0", + "eslint": "^9.33.0", + "eslint-plugin-react-hooks": "^5.2.0", + "eslint-plugin-react-refresh": "^0.4.20", + "globals": "^16.3.0", + "vite": "^7.1.2" + } +} diff --git a/frontend/public/LOGO.png b/frontend/public/LOGO.png new file mode 100644 index 0000000..16a1786 Binary files /dev/null and b/frontend/public/LOGO.png differ diff --git a/frontend/public/static/css/style.css b/frontend/public/static/css/style.css new file mode 100644 index 0000000..3025264 --- /dev/null +++ b/frontend/public/static/css/style.css @@ -0,0 +1,97 @@ +/* Dark Theme Adaptation for tools.html */ +body { + font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif; + background-color: #121212; /* Match MUI dark theme background */ + color: #e0e0e0; /* Light text color for dark background */ + margin: 20px; + line-height: 1.6; +} + +h1, h2 { + color: #ffffff; + border-bottom: 2px solid #424242; /* Darker border */ + padding-bottom: 10px; +} + +.container { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(400px, 1fr)); + gap: 20px; +} + +.card { + background-color: #1e1e1e; /* Darker card background */ + border: 1px solid #424242; /* Subtle border */ + border-radius: 8px; + padding: 20px; + box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2); +} + +.card h3 { + margin-top: 0; + color: #bb86fc; /* A nice accent color for dark theme */ +} + +input[type="file"], textarea { + width: 100%; + padding: 8px; + margin-top: 10px; + border-radius: 4px; + border: 1px solid #555; + background-color: #333; + color: #e0e0e0; +} + +button { + background-color: #3700b3; /* MUI dark theme primary-like color */ + color: white; + padding: 10px 15px; + border: none; + border-radius: 4px; + cursor: pointer; + margin-top: 10px; + transition: background-color 0.3s; +} + +button:hover { + background-color: #6200ee; +} + +button:disabled { + background-color: #444; + cursor: not-allowed; +} + +.progress-container, .result-container { + margin-top: 15px; + background-color: #2c2c2c; + padding: 15px; + border-radius: 4px; +} + +.progress-bar { + width: 100%; + background-color: #444; + border-radius: 4px; + overflow: hidden; +} + +.progress-bar-inner { + height: 20px; + width: 0%; + background-color: #03dac6; /* Accent color for progress */ + text-align: center; + line-height: 20px; + color: black; + transition: width 0.4s ease; +} + +#translation-preview { + white-space: pre-wrap; + max-height: 200px; + overflow-y: auto; + border: 1px solid #555; + padding: 10px; + margin-top: 10px; + background-color: #333; +} diff --git a/frontend/public/vite.svg b/frontend/public/vite.svg new file mode 100644 index 0000000..e7b8dfb --- /dev/null +++ b/frontend/public/vite.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/frontend/src/App.css b/frontend/src/App.css new file mode 100644 index 0000000..b9d355d --- /dev/null +++ b/frontend/src/App.css @@ -0,0 +1,42 @@ +#root { + max-width: 1280px; + margin: 0 auto; + padding: 2rem; + text-align: center; +} + +.logo { + height: 6em; + padding: 1.5em; + will-change: filter; + transition: filter 300ms; +} +.logo:hover { + filter: drop-shadow(0 0 2em #646cffaa); +} +.logo.react:hover { + filter: drop-shadow(0 0 2em #61dafbaa); +} + +@keyframes logo-spin { + from { + transform: rotate(0deg); + } + to { + transform: rotate(360deg); + } +} + +@media (prefers-reduced-motion: no-preference) { + a:nth-of-type(2) .logo { + animation: logo-spin infinite 20s linear; + } +} + +.card { + padding: 2em; +} + +.read-the-docs { + color: #888; +} diff --git a/frontend/src/App.jsx b/frontend/src/App.jsx new file mode 100644 index 0000000..50998aa --- /dev/null +++ b/frontend/src/App.jsx @@ -0,0 +1,68 @@ +import React from 'react'; +import { Routes, Route, Navigate, Outlet } from 'react-router-dom'; +import { useAuth } from './contexts/AuthContext'; +import LoginPage from './pages/LoginPage'; +import DashboardPage from './pages/DashboardPage'; +import MeetingDetailPage from './pages/MeetingDetailPage'; +import ProcessingPage from './pages/ProcessingPage'; // Restored +import AdminPage from './pages/AdminPage'; // Added +import Layout from './components/Layout'; +import { ThemeProvider, createTheme } from '@mui/material/styles'; +import CssBaseline from '@mui/material/CssBaseline'; +import { CircularProgress, Box } from '@mui/material'; + +const darkTheme = createTheme({ + palette: { + mode: 'dark', + }, +}); + +const PrivateRoute = () => { + const { user, loading } = useAuth(); + + if (loading) { + return ( + + + + ); + } + + return user ? : ; +}; + +const AdminRoute = () => { + const { user, loading } = useAuth(); + + if (loading) { + return ( + + + + ); + } + + return user && user.role === 'admin' ? : ; +}; + +function App() { + return ( + + + + } /> + }> + } /> + } /> + } /> + + }> + } /> + + + + + ); +} + +export default App; diff --git a/frontend/src/assets/react.svg b/frontend/src/assets/react.svg new file mode 100644 index 0000000..6c87de9 --- /dev/null +++ b/frontend/src/assets/react.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/frontend/src/components/Layout.jsx b/frontend/src/components/Layout.jsx new file mode 100644 index 0000000..f8065e8 --- /dev/null +++ b/frontend/src/components/Layout.jsx @@ -0,0 +1,52 @@ +import React from 'react'; +import { AppBar, Toolbar, Typography, Button, Container, Box } from '@mui/material'; +import { useAuth } from '../contexts/AuthContext'; +import { Link, useNavigate } from 'react-router-dom'; + +const Layout = ({ children }) => { + const { user, logout } = useAuth(); + const navigate = useNavigate(); + + const handleLogout = () => { + logout(); + navigate('/login'); + }; + + return ( + + + + + AI Meeting Assistant + + {user && ( + + + + {user.role === 'admin' && ( + + )} + + | Welcome, {user.username} ({user.role}) + + + + )} + + + + {children} + + + ); +}; + +export default Layout; diff --git a/frontend/src/components/NewMeetingDialog.jsx b/frontend/src/components/NewMeetingDialog.jsx new file mode 100644 index 0000000..a73a81a --- /dev/null +++ b/frontend/src/components/NewMeetingDialog.jsx @@ -0,0 +1,82 @@ +import React, { useState } from 'react'; +import { + Button, TextField, Dialog, DialogTitle, DialogContent, DialogActions, + CircularProgress, Alert +} from '@mui/material'; + +const NewMeetingDialog = ({ open, onClose, onCreate }) => { + const [topic, setTopic] = useState(''); + const [meetingDate, setMeetingDate] = useState(new Date().toISOString().split('T')[0]); // Defaults to today + const [loading, setLoading] = useState(false); + const [error, setError] = useState(''); + + const handleCreate = async () => { + if (!topic || !meetingDate) { + setError('Both topic and date are required.'); + return; + } + setError(''); + setLoading(true); + try { + // The onCreate prop is expected to be an async function + // that returns the newly created meeting. + await onCreate(topic, meetingDate); + handleClose(); // Close the dialog on success + } catch (err) { + setError(err.message || 'Failed to create meeting.'); + } finally { + setLoading(false); + } + }; + + const handleClose = () => { + if (loading) return; // Prevent closing while loading + setTopic(''); + setMeetingDate(new Date().toISOString().split('T')[0]); + setError(''); + onClose(); // Call the parent's onClose handler + }; + + return ( + + Create a New Meeting + + {error && {error}} + setTopic(e.target.value)} + sx={{ mt: 2 }} + /> + setMeetingDate(e.target.value)} + InputLabelProps={{ + shrink: true, + }} + sx={{ mt: 2 }} + /> + + + + + + + ); +}; + +export default NewMeetingDialog; diff --git a/frontend/src/components/TextProcessingTools.jsx b/frontend/src/components/TextProcessingTools.jsx new file mode 100644 index 0000000..d158439 --- /dev/null +++ b/frontend/src/components/TextProcessingTools.jsx @@ -0,0 +1,116 @@ +import React, { useState, useEffect } from 'react'; +import { + Box, Button, Typography, Paper, TextField, Select, MenuItem, FormControl, InputLabel, Dialog, DialogTitle, DialogContent, DialogActions, CircularProgress, Alert, Table, TableBody, TableCell, TableContainer, TableHead, TableRow +} from '@mui/material'; +import { getMeetings, createMeeting, batchCreateActionItems } from '../services/api'; // Only necessary APIs + +const TextProcessingTools = ({ + textContent, + summary, + actionItems, + onGenerateSummary, + onPreviewActions, + onActionItemChange +}) => { + const [meetings, setMeetings] = useState([]); + const [users, setUsers] = useState([]); // Assuming users are needed for dropdown + const [isMeetingDialogOpen, setIsMeetingDialogOpen] = useState(false); + const [associationType, setAssociationType] = useState('existing'); + const [selectedMeetingId, setSelectedMeetingId] = useState(''); + const [newMeetingTopic, setNewMeetingTopic] = useState(''); + const [saveLoading, setSaveLoading] = useState(false); + const [error, setError] = useState(''); + + useEffect(() => { + const fetchDropdownData = async () => { + try { + const meetingsRes = await getMeetings(); + setMeetings(meetingsRes.data); + if (meetingsRes.data.length > 0) { + setSelectedMeetingId(meetingsRes.data[0].id); + } + } catch (err) { console.error('Could not fetch meetings for dropdown.'); } + }; + fetchDropdownData(); + }, []); + + const handleInitiateSave = () => { + if (!actionItems || !Array.isArray(actionItems) || actionItems.length === 0) { + setError('No valid action items to save.'); + return; + } + setError(''); + setIsMeetingDialogOpen(true); + }; + + const handleConfirmSave = async () => { + let meetingIdToSave = selectedMeetingId; + if (associationType === 'new') { + if (!newMeetingTopic) { setError('New meeting topic is required.'); return; } + try { + const { data: newMeeting } = await createMeeting(newMeetingTopic, new Date().toISOString()); + meetingIdToSave = newMeeting.id; + } catch (err) { setError('Failed to create new meeting.'); return; } + } + + if (!meetingIdToSave) { setError('A meeting must be selected or created.'); return; } + + setSaveLoading(true); setError(''); + try { + const itemsToSave = actionItems.map(({ tempId, owner, duedate, ...rest }) => rest); + await batchCreateActionItems(meetingIdToSave, itemsToSave); + setIsMeetingDialogOpen(false); + alert('Action Items saved successfully!'); + // Optionally, clear items after save by calling a prop function from parent + } catch (err) { setError(err.response?.data?.error || 'Failed to save action items.'); } + finally { setSaveLoading(false); } + }; + + return ( + + {error && {error}} + + + + + + {summary && Summary} + + {actionItems && actionItems.length > 0 && ( + + Review and Edit Action Items + + + ContextActionOwnerDue Date + {actionItems.map(item => ( + + onActionItemChange(item.tempId, 'item', e.target.value)}/> + onActionItemChange(item.tempId, 'action', e.target.value)}/> + onActionItemChange(item.tempId, 'owner', e.target.value)}/> + onActionItemChange(item.tempId, 'due_date', e.target.value)} InputLabelProps={{ shrink: true }}/> + + ))} +
+
+ + + +
+ )} + + setIsMeetingDialogOpen(false)} fullWidth maxWidth="xs"> + Associate with a Meeting + + + {associationType === 'existing' ? Select Meeting : setNewMeetingTopic(e.target.value)} />} + + + + + + +
+ ); +}; + +export default TextProcessingTools; diff --git a/frontend/src/contexts/AuthContext.jsx b/frontend/src/contexts/AuthContext.jsx new file mode 100644 index 0000000..3026768 --- /dev/null +++ b/frontend/src/contexts/AuthContext.jsx @@ -0,0 +1,88 @@ +import React, { createContext, useState, useContext, useEffect } from 'react'; +import axios from 'axios'; +import { jwtDecode } from 'jwt-decode'; + +const AuthContext = createContext(null); + +const setAuthToken = token => { + if (token) { + axios.defaults.headers.common['Authorization'] = `Bearer ${token}`; + } else { + delete axios.defaults.headers.common['Authorization']; + } +}; + +export const AuthProvider = ({ children }) => { + const [user, setUser] = useState(null); + const [token, setToken] = useState(() => localStorage.getItem('token')); + const [loading, setLoading] = useState(true); + + useEffect(() => { + if (token) { + try { + const decoded = jwtDecode(token); + const currentTime = Date.now() / 1000; + if (decoded.exp < currentTime) { + console.log("Token expired, logging out."); + logout(); + } else { + setUser({ + id: decoded.sub, + role: decoded.role, + username: decoded.username +}); + setAuthToken(token); + } + } catch (error) { + console.error("Invalid token on initial load"); + logout(); + } + } + setLoading(false); + }, [token]); + + const login = async (username, password) => { + try { + const response = await axios.post('/api/login', { username, password }); + const { access_token } = response.data; + localStorage.setItem('token', access_token); + setToken(access_token); + const decoded = jwtDecode(access_token); + setUser({ + id: decoded.sub, + role: decoded.role, + username: decoded.username +}); + setAuthToken(access_token); + return { success: true }; + } catch (error) { + console.error('Login failed:', error.response?.data?.msg || error.message); + return { success: false, message: error.response?.data?.msg || 'Login failed' }; + } + }; + + const logout = () => { + localStorage.removeItem('token'); + setToken(null); + setUser(null); + setAuthToken(null); + }; + + const value = { + user, + token, + loading, + login, + logout + }; + + return ( + + {!loading && children} + + ); +}; + +export const useAuth = () => { + return useContext(AuthContext); +}; diff --git a/frontend/src/index.css b/frontend/src/index.css new file mode 100644 index 0000000..e9a74c8 --- /dev/null +++ b/frontend/src/index.css @@ -0,0 +1,27 @@ +/* Reset default styles and ensure full-width/height layout */ +html, body, #root { + margin: 0; + padding: 0; + width: 100%; + height: 100%; +} + +body { + font-family: system-ui, Avenir, Helvetica, Arial, sans-serif; + line-height: 1.5; + font-weight: 400; + font-synthesis: none; + text-rendering: optimizeLegibility; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +/* The rest of the default styles can be kept or removed as needed */ +a { + font-weight: 500; + color: #646cff; + text-decoration: inherit; +} +a:hover { + color: #535bf2; +} diff --git a/frontend/src/main.jsx b/frontend/src/main.jsx new file mode 100644 index 0000000..c16bff3 --- /dev/null +++ b/frontend/src/main.jsx @@ -0,0 +1,16 @@ +import React from 'react' +import ReactDOM from 'react-dom/client' +import App from './App.jsx' +import './index.css' +import { BrowserRouter } from 'react-router-dom'; +import { AuthProvider } from './contexts/AuthContext.jsx'; + +ReactDOM.createRoot(document.getElementById('root')).render( + + + + + + + , +) diff --git a/frontend/src/pages/ActionItemPage.jsx b/frontend/src/pages/ActionItemPage.jsx new file mode 100644 index 0000000..0f5b6b2 --- /dev/null +++ b/frontend/src/pages/ActionItemPage.jsx @@ -0,0 +1,174 @@ +import React, { useState, useEffect } from 'react'; +import { useParams, useNavigate } from 'react-router-dom'; +import { + Box, Typography, Paper, CircularProgress, Alert, + TextField, Button, Select, MenuItem, FormControl, InputLabel, Grid, Link +} from '@mui/material'; +import { getActionItemDetails, updateActionItem, uploadAttachment } from '../services/api'; + +const ActionItemPage = () => { + const { actionId } = useParams(); + const navigate = useNavigate(); + const [actionItem, setActionItem] = useState(null); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(''); + const [isEditing, setIsEditing] = useState(false); + const [attachment, setAttachment] = useState(null); + + useEffect(() => { + const fetchActionItem = async () => { + setLoading(true); + try { + const data = await getActionItemDetails(actionId); + setActionItem(data); + } catch (err) { + setError(err.message || 'Could not fetch action item details.'); + } finally { + setLoading(false); + } + }; + fetchActionItem(); + }, [actionId]); + + const handleUpdate = async () => { + if (!actionItem) return; + setLoading(true); + try { + // Only send fields that are meant to be updated + const updateData = { + item: actionItem.item, + action: actionItem.action, + status: actionItem.status, + due_date: actionItem.due_date, + // owner_id is typically not changed from this screen, but could be added if needed + }; + await updateActionItem(actionId, updateData); + + if (attachment) { + // Note: The backend needs an endpoint to handle attachment uploads for an action item. + // This is a placeholder for that functionality. + // await uploadAttachment(actionId, attachment); + console.warn("Attachment upload functionality is not yet implemented on the backend."); + } + + setIsEditing(false); + // Refresh data after update + const data = await getActionItemDetails(actionId); + setActionItem(data); + + } catch (err) { + setError(err.response?.data?.error || 'Failed to update action item.'); + } finally { + setLoading(false); + } + }; + + const handleFileChange = (event) => { + setAttachment(event.target.files[0]); + }; + + if (loading) return ; + if (error) return {error}; + if (!actionItem) return No action item found.; + + return ( + + + Action Item for: {actionItem.meeting?.topic || 'General Task'} + + + + + setActionItem({ ...actionItem, item: e.target.value })} + InputProps={{ readOnly: !isEditing }} + variant={isEditing ? "outlined" : "filled"} + /> + + + setActionItem({ ...actionItem, action: e.target.value })} + InputProps={{ readOnly: !isEditing }} + variant={isEditing ? "outlined" : "filled"} + /> + + + + Status + + + + + setActionItem({ ...actionItem, due_date: e.target.value })} + InputProps={{ readOnly: !isEditing }} + variant={isEditing ? "outlined" : "filled"} + InputLabelProps={{ shrink: true }} + /> + + + Owner: {actionItem.owner?.username || 'N/A'} + {actionItem.attachment_path && ( + + Attachment: Download + + )} + + {isEditing && ( + + + {attachment && {attachment.name}} + + )} + + {isEditing ? ( + + + + + ) : ( + + )} + + + + + ); +}; + +export default ActionItemPage; \ No newline at end of file diff --git a/frontend/src/pages/AdminPage.jsx b/frontend/src/pages/AdminPage.jsx new file mode 100644 index 0000000..c209664 --- /dev/null +++ b/frontend/src/pages/AdminPage.jsx @@ -0,0 +1,237 @@ +import React, { useState, useEffect, useCallback } from 'react'; +import { + Box, Typography, Paper, Table, TableBody, TableCell, TableContainer, TableHead, TableRow, + CircularProgress, Alert, Button, Dialog, DialogActions, DialogContent, DialogTitle, TextField, + Select, MenuItem, FormControl, InputLabel, IconButton, Tooltip +} from '@mui/material'; +import AddCircleOutlineIcon from '@mui/icons-material/AddCircleOutline'; +import DeleteIcon from '@mui/icons-material/Delete'; +import LockResetIcon from '@mui/icons-material/LockReset'; +import { getUsers, createUser, deleteUser, changeUserPassword } from '../services/api'; +import { useAuth } from '../contexts/AuthContext'; + +const PasswordChangeDialog = ({ open, onClose, onConfirm, user }) => { + const [password, setPassword] = useState(''); + const [confirmPassword, setConfirmPassword] = useState(''); + const [error, setError] = useState(''); + + const handleConfirm = () => { + if (!password) { + setError('Password cannot be empty.'); + return; + } + if (password !== confirmPassword) { + setError('Passwords do not match.'); + return; + } + onConfirm(user.id, password); + }; + + const handleClose = () => { + setPassword(''); + setConfirmPassword(''); + setError(''); + onClose(); + }; + + return ( + + Change Password for {user?.username} + + {error && {error}} + setPassword(e.target.value)} /> + setConfirmPassword(e.target.value)} /> + + + + + + + ); +}; + + +const AdminPage = () => { + const { user: currentUser } = useAuth(); + const [users, setUsers] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(''); + const [success, setSuccess] = useState(''); + + const [isUserDialogOpen, setIsUserDialogOpen] = useState(false); + const [newUser, setNewUser] = useState({ username: '', password: '', role: 'user' }); + + const [isPasswordDialogOpen, setIsPasswordDialogOpen] = useState(false); + const [selectedUser, setSelectedUser] = useState(null); + + + const fetchUsers = useCallback(async () => { + try { + setLoading(true); + const data = await getUsers(); + setUsers(data); + setError(''); + } catch (err) { + setError(err.response?.data?.msg || 'Could not fetch users.'); + } finally { + setLoading(false); + } + }, []); + + useEffect(() => { + fetchUsers(); + }, [fetchUsers]); + + const handleOpenUserDialog = () => setIsUserDialogOpen(true); + const handleCloseUserDialog = () => { + setIsUserDialogOpen(false); + setNewUser({ username: '', password: '', role: 'user' }); // Reset form + }; + + const handleOpenPasswordDialog = (user) => { + setSelectedUser(user); + setIsPasswordDialogOpen(true); + }; + const handleClosePasswordDialog = () => { + setSelectedUser(null); + setIsPasswordDialogOpen(false); + }; + + const handleCreateUser = async () => { + if (!newUser.username || !newUser.password) { + setError('Username and password are required.'); + return; + } + try { + await createUser(newUser); + handleCloseUserDialog(); + fetchUsers(); // Refetch the list + setSuccess('User created successfully.'); + } catch (err) { + setError(err.response?.data?.error || 'Failed to create user.'); + } + }; + + const handleDeleteUser = async (userId) => { + if (window.confirm('Are you sure you want to delete this user? This will disassociate them from all meetings and action items.')) { + try { + await deleteUser(userId); + fetchUsers(); // Refresh the list + setSuccess('User deleted successfully.'); + } catch (err) { + setError(err.response?.data?.error || 'Failed to delete user.'); + } + } + }; + + const handlePasswordChange = async (userId, newPassword) => { + try { + await changeUserPassword(userId, newPassword); + handleClosePasswordDialog(); + setSuccess('Password updated successfully.'); + } catch (err) { + // The dialog will show specific errors, this is a fallback. + setError(err.response?.data?.error || 'Failed to change password.'); + } + }; + + const handleInputChange = (e) => { + const { name, value } = e.target; + setNewUser(prev => ({ ...prev, [name]: value })); + }; + + if (loading) { + return ; + } + + return ( + + + + User Management + + + + + {error && setError('')}>{error}} + {success && setSuccess('')}>{success}} + + + + + + + ID + Username + Role + Created At + Actions + + + + {users.map((user) => ( + + {user.id} + {user.username} + {user.role} + {new Date(user.created_at).toLocaleString()} + + + handleOpenPasswordDialog(user)}> + + + + {/* Prevent admin from deleting themselves, show placeholder otherwise */} + {String(currentUser.id) !== String(user.id) ? ( + + handleDeleteUser(user.id)}> + + + + ) : ( + // Invisible placeholder to maintain alignment. An IconButton is ~40px wide. + + )} + + + ))} + +
+
+ + {/* Create User Dialog */} + + Create a New User + + + + + Role + + + + + + + + + + {/* Change Password Dialog */} + {selectedUser && ( + + )} +
+ ); +}; + +export default AdminPage; \ No newline at end of file diff --git a/frontend/src/pages/DashboardPage.jsx b/frontend/src/pages/DashboardPage.jsx new file mode 100644 index 0000000..12034ca --- /dev/null +++ b/frontend/src/pages/DashboardPage.jsx @@ -0,0 +1,291 @@ +import React, { useState, useEffect, useMemo, useCallback } from 'react'; +import { useNavigate } from 'react-router-dom'; +import { + Box, Typography, Paper, Table, TableBody, TableCell, TableContainer, TableHead, TableRow, + CircularProgress, Alert, Button, TextField, TableSortLabel, Select, MenuItem, FormControl, + InputLabel, Chip, OutlinedInput, IconButton +} from '@mui/material'; +import AddCircleOutlineIcon from '@mui/icons-material/AddCircleOutline'; +import DeleteIcon from '@mui/icons-material/Delete'; +import { getMeetings, updateMeeting, deleteMeeting, createMeeting } from '../services/api'; +import { useAuth } from '../contexts/AuthContext'; +import NewMeetingDialog from '../components/NewMeetingDialog'; + +// Helper function for sorting +function descendingComparator(a, b, orderBy) { + if (b[orderBy] < a[orderBy]) return -1; + if (b[orderBy] > a[orderBy]) return 1; + return 0; +} + +function getComparator(order, orderBy) { + return order === 'desc' + ? (a, b) => descendingComparator(a, b, orderBy) + : (a, b) => -descendingComparator(a, b, orderBy); +} + +const DashboardPage = () => { + const navigate = useNavigate(); + const { user } = useAuth(); + const [meetings, setMeetings] = useState([]); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(''); + const [deletingIds, setDeletingIds] = useState(new Set()); + const [isModalOpen, setIsModalOpen] = useState(false); + + // State for filtering and searching + const [topicSearch, setTopicSearch] = useState(''); + const [ownerSearch, setOwnerSearch] = useState(''); + const [statusFilter, setStatusFilter] = useState([]); + + // State for sorting + const [order, setOrder] = useState('asc'); + const [orderBy, setOrderBy] = useState('meeting_date'); + + const fetchMeetings = useCallback(async () => { + try { + const data = await getMeetings(); + setMeetings(data); + } catch (err) { + setError('Could not fetch meetings.'); + } finally { + setLoading(false); + } + }, []); + + useEffect(() => { + fetchMeetings(); + }, [fetchMeetings]); + + const handleCreateMeeting = async (topic, meetingDate) => { + try { + const newMeeting = await createMeeting(topic, new Date(meetingDate).toISOString()); + setIsModalOpen(false); + navigate(`/meeting/${newMeeting.id}`); + } catch (err) { + throw new Error(err.response?.data?.error || "Failed to create meeting."); + } + }; + + const handleSortRequest = (property) => { + const isAsc = orderBy === property && order === 'asc'; + setOrder(isAsc ? 'desc' : 'asc'); + setOrderBy(property); + }; + + const handleStatusChange = async (meetingId, newStatus) => { + try { + await updateMeeting(meetingId, { status: newStatus }); + fetchMeetings(); + } catch (err) { + setError(`Failed to update status for meeting ${meetingId}.`); + } + }; + + const handleDelete = async (meetingId) => { + if (window.confirm('Are you sure you want to delete this meeting?')) { + setDeletingIds(prev => new Set(prev).add(meetingId)); + try { + await deleteMeeting(meetingId); + fetchMeetings(); + } catch (err) { + setError(`Failed to delete meeting ${meetingId}.`); + } finally { + setDeletingIds(prev => { + const newSet = new Set(prev); + newSet.delete(meetingId); + return newSet; + }); + } + } + }; + + const uniqueStatuses = useMemo(() => { + const statuses = new Set(meetings.map(m => m.status)); + return Array.from(statuses); + }, [meetings]); + + const filteredAndSortedMeetings = useMemo(() => { + let filtered = meetings.filter(meeting => { + const topicMatch = meeting.topic.toLowerCase().includes(topicSearch.toLowerCase()); + const ownerMatch = meeting.owner_name ? meeting.owner_name.toLowerCase().includes(ownerSearch.toLowerCase()) : ownerSearch === ''; + const statusMatch = statusFilter.length === 0 || statusFilter.includes(meeting.status); + return topicMatch && ownerMatch && statusMatch; + }); + + return filtered.sort(getComparator(order, orderBy)); + }, [meetings, topicSearch, ownerSearch, statusFilter, order, orderBy]); + + if (loading) { + return ; + } + + const headCells = [ + { id: 'topic', label: 'Topic' }, + { id: 'owner_name', label: 'Owner' }, + { id: 'meeting_date', label: 'Meeting Date' }, + { id: 'status', label: 'Status' }, + { id: 'action_item_count', label: 'Action Items' }, + { id: 'created_at', label: 'Created At' }, + ]; + + const statusColorMap = { + 'Completed': 'success', + 'In Progress': 'info', + 'To Do': 'warning', + 'Failed': 'error', + }; + + const allPossibleStatuses = ['To Do', 'In Progress', 'Completed', 'Failed']; + + return ( + + + Dashboard + + + + setIsModalOpen(false)} + onCreate={handleCreateMeeting} + /> + + {error && setError('')}>{error}} + + + + setTopicSearch(e.target.value)} + sx={{ flex: '1 1 40%' }} + /> + setOwnerSearch(e.target.value)} + sx={{ flex: '1 1 30%' }} + /> + + Filter by Status + + + + + + + + + + {headCells.map((headCell) => ( + + handleSortRequest(headCell.id)} + > + {headCell.label} + + + ))} + Actions + + + + {filteredAndSortedMeetings.map((meeting) => { + const isOwnerOrAdmin = user && (user.role === 'admin' || String(user.id) === String(meeting.created_by_id)); + const isDeleting = deletingIds.has(meeting.id); + + let taipeiTime = 'N/A'; + if (meeting.created_at) { + taipeiTime = new Date(meeting.created_at).toLocaleString('zh-TW', { + timeZone: 'Asia/Taipei', + year: 'numeric', + month: '2-digit', + day: '2-digit', + hour: '2-digit', + minute: '2-digit', + second: '2-digit', + hour12: false + }); + } + + return ( + + {meeting.topic} + {meeting.owner_name || 'N/A'} + {meeting.meeting_date ? new Date(meeting.meeting_date).toLocaleDateString() : 'N/A'} + + + + + + {meeting.action_item_count} + {taipeiTime} + + + + {isOwnerOrAdmin ? ( + handleDelete(meeting.id)} color="error" disabled={isDeleting}> + {isDeleting ? : } + + ) : ( + // Invisible placeholder to maintain alignment + + )} + + + + ); + })} + +
+
+
+ ); +}; + +export default DashboardPage; diff --git a/frontend/src/pages/LoginPage.jsx b/frontend/src/pages/LoginPage.jsx new file mode 100644 index 0000000..a00d87d --- /dev/null +++ b/frontend/src/pages/LoginPage.jsx @@ -0,0 +1,155 @@ +import React, { useState } from 'react'; +import { useAuth } from '../contexts/AuthContext'; +import { useNavigate, useLocation } from 'react-router-dom'; +import { + Button, TextField, Container, Typography, Box, Alert, Grid, Link, Avatar, Card, CardContent, CircularProgress +} from '@mui/material'; +import LockOutlinedIcon from '@mui/icons-material/LockOutlined'; +import { register } from '../services/api'; + +const LoginPage = () => { + const [username, setUsername] = useState(''); + const [password, setPassword] = useState(''); + const [confirmPassword, setConfirmPassword] = useState(''); + const [error, setError] = useState(''); + const [success, setSuccess] = useState(''); + const [loading, setLoading] = useState(false); + const [isRegister, setIsRegister] = useState(false); + + const { login } = useAuth(); + const navigate = useNavigate(); + const location = useLocation(); + + const from = location.state?.from?.pathname || "/"; + + const handleLogin = async (e) => { + e.preventDefault(); + setError(''); + setSuccess(''); + setLoading(true); + const { success, message } = await login(username, password); + if (success) { + navigate(from, { replace: true }); + } else { + setError(message || 'Failed to log in'); + } + setLoading(false); + }; + + const handleRegister = async (e) => { + e.preventDefault(); + if (password !== confirmPassword) { + setError('Passwords do not match.'); + return; + } + setError(''); + setSuccess(''); + setLoading(true); + try { + await register(username, password); + setSuccess('Account created successfully! Please log in.'); + setIsRegister(false); // Switch back to login view + setUsername(''); // Clear fields + setPassword(''); + setConfirmPassword(''); + } catch (err) { + setError(err.response?.data?.error || 'Failed to create account.'); + } + setLoading(false); + }; + + return ( + + + + + + + + + AI Meeting Assistant + + + {isRegister ? 'Create Account' : 'Sign In'} + + + setUsername(e.target.value)} + /> + setPassword(e.target.value)} + /> + {isRegister && ( + setConfirmPassword(e.target.value)} + /> + )} + {error && {error}} + {success && {success}} + + + + { + setIsRegister(!isRegister); + setError(''); + setSuccess(''); + }}> + {isRegister ? "Already have an account? Sign In" : "Don't have an account? Sign Up"} + + + + + + + + + ); +}; + +export default LoginPage; \ No newline at end of file diff --git a/frontend/src/pages/MeetingDetailPage.jsx b/frontend/src/pages/MeetingDetailPage.jsx new file mode 100644 index 0000000..9b1e416 --- /dev/null +++ b/frontend/src/pages/MeetingDetailPage.jsx @@ -0,0 +1,308 @@ +import React, { useState, useEffect, useCallback } from 'react'; +import { useParams } from 'react-router-dom'; +import { + Box, Typography, Paper, CircularProgress, Alert, Button, IconButton, Dialog, DialogTitle, DialogContent, DialogActions, + Table, TableBody, TableCell, TableContainer, TableHead, TableRow, TextField, Select, MenuItem, FormControl, InputLabel, + Grid, Card, CardContent +} from '@mui/material'; +import AddCircleOutlineIcon from '@mui/icons-material/AddCircleOutline'; +import EditIcon from '@mui/icons-material/Edit'; +import DeleteIcon from '@mui/icons-material/Delete'; +import SaveIcon from '@mui/icons-material/Save'; +import CancelIcon from '@mui/icons-material/Cancel'; +import PreviewIcon from '@mui/icons-material/Preview'; +import DownloadIcon from '@mui/icons-material/Download'; +import { + getMeetingDetails, updateMeeting, summarizeMeeting, + getActionItemsForMeeting, createActionItem, updateActionItem, deleteActionItem, getAllUsers, + previewActionItems, batchSaveActionItems, pollTaskStatus, uploadActionItemAttachment, downloadFile +} from '../services/api'; +import { useAuth } from '../contexts/AuthContext'; + +const MeetingDetailPage = () => { + const { meetingId } = useParams(); + const { user: currentUser } = useAuth(); + + const [meeting, setMeeting] = useState(null); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(''); + const [isEditingTranscript, setIsEditingTranscript] = useState(false); + const [isEditingSummary, setIsEditingSummary] = useState(false); + const [editData, setEditData] = useState({}); + const [summaryTask, setSummaryTask] = useState(null); + + const [actionItems, setActionItems] = useState([]); + const [users, setUsers] = useState([]); + const [editingActionItemId, setEditingActionItemId] = useState(null); + const [editActionItemData, setEditActionItemData] = useState({}); + const [attachmentFile, setAttachmentFile] = useState(null); + const [isAddActionItemOpen, setIsAddActionItemOpen] = useState(false); + const [newActionItem, setNewActionItem] = useState({ action: '', owner_id: '', due_date: '', item: '' }); + const [previewedItems, setPreviewedItems] = useState([]); + const [isPreviewLoading, setIsPreviewLoading] = useState(false); + + const fetchMeetingData = useCallback(async () => { + try { + setLoading(true); + const meetingRes = await getMeetingDetails(meetingId); + setMeeting(meetingRes); + setEditData(meetingRes); + const itemsRes = await getActionItemsForMeeting(meetingId); + setActionItems(itemsRes); + } catch (err) { + setError('Failed to fetch meeting data.'); + } finally { + setLoading(false); + } + }, [meetingId]); + + useEffect(() => { + fetchMeetingData(); + }, [fetchMeetingData]); + + useEffect(() => { + const fetchUsers = async () => { + try { + const usersRes = await getAllUsers(); + setUsers(usersRes); + } catch (err) { console.warn('Could not fetch user list.'); } + }; + fetchUsers(); + }, []); + + useEffect(() => { + let intervalId = null; + if (summaryTask && (summaryTask.state === 'PENDING' || summaryTask.state === 'PROGRESS')) { + intervalId = setInterval(async () => { + try { + const updatedTask = await pollTaskStatus(summaryTask.status_url); + if (['SUCCESS', 'FAILURE', 'REVOKED'].includes(updatedTask.state)) { + clearInterval(intervalId); + setSummaryTask(null); + if (updatedTask.state === 'SUCCESS' && updatedTask.info.summary) { + // Directly update the summary instead of refetching everything + setMeeting(prevMeeting => ({...prevMeeting, summary: updatedTask.info.summary})); + setEditData(prevEditData => ({...prevEditData, summary: updatedTask.info.summary})); + } else { + // Fallback to refetch if something goes wrong or task fails + fetchMeetingData(); + } + } + } catch (err) { + console.error('Polling failed:', err); + clearInterval(intervalId); + setSummaryTask(null); + } + }, 2000); + } + return () => clearInterval(intervalId); + }, [summaryTask, fetchMeetingData]); + + const handleSave = async (field, value) => { + try { + await updateMeeting(meetingId, { [field]: value }); + fetchMeetingData(); + return true; + } catch (err) { + setError(`Failed to save ${field}.`); + return false; + } + }; + + const handleSaveTranscript = async () => { + if (await handleSave('transcript', editData.transcript)) setIsEditingTranscript(false); + }; + + const handleSaveSummary = async () => { + if (await handleSave('summary', editData.summary)) setIsEditingSummary(false); + }; + + const handleGenerateSummary = async () => { + try { + const taskInfo = await summarizeMeeting(meetingId); + setSummaryTask({ ...taskInfo, state: 'PENDING' }); + } catch (err) { + setError('Failed to start summary generation.'); + } + }; + + const handlePreviewActionItems = async () => { + const textToPreview = meeting?.summary || meeting?.transcript; + if (!textToPreview) return; + setIsPreviewLoading(true); + try { + const result = await previewActionItems(textToPreview); + setPreviewedItems(result.items || []); + } catch (err) { + setError('Failed to generate action item preview.'); + } finally { + setIsPreviewLoading(false); + } + }; + + const handleBatchSave = async () => { + if (previewedItems.length === 0) return; + try { + await batchSaveActionItems(meetingId, previewedItems); + setPreviewedItems([]); + fetchMeetingData(); + } catch (err) { + setError('Failed to save action items.'); + } + }; + + const handleEditActionItemClick = (item) => { setEditingActionItemId(item.id); setEditActionItemData({ ...item, due_date: item.due_date || '' }); }; + const handleCancelActionItemClick = () => { setEditingActionItemId(null); setAttachmentFile(null); }; + const handleSaveActionItemClick = async (id) => { + try { + await updateActionItem(id, editActionItemData); + if (attachmentFile) await uploadActionItemAttachment(id, attachmentFile); + setEditingActionItemId(null); + setAttachmentFile(null); + fetchMeetingData(); + } catch (err) { + setError('Failed to save action item.'); + } + }; + const handleDeleteActionItemClick = async (id) => { if (window.confirm('Are you sure?')) { try { await deleteActionItem(id); fetchMeetingData(); } catch (err) { setError('Failed to delete action item.'); }}}; + const handleAddActionItemSave = async () => { if (!newActionItem.action) { setError('Action is required.'); return; } try { const newItem = await createActionItem({ ...newActionItem, meeting_id: meetingId }); if (attachmentFile) { await uploadActionItemAttachment(newItem.id, attachmentFile); } setIsAddActionItemOpen(false); setNewActionItem({ action: '', owner_id: '', due_date: '', item: '' }); setAttachmentFile(null); fetchMeetingData(); } catch (err) { setError('Failed to create action item.'); }}; + const handleFileChange = (e) => { if (e.target.files[0]) setAttachmentFile(e.target.files[0]); }; + + if (loading) return ; + if (!meeting) return Meeting not found.; + + const canManageMeeting = currentUser && meeting && (currentUser.role === 'admin' || String(currentUser.id) === String(meeting.created_by_id)); + + return ( + + {error && {error}} + + {/* Transcript Card (Full Width) */} + + + + {isEditingTranscript ? ( + <> + setEditData({...editData, transcript: e.target.value})} /> + + + ) : ( + <> + + {meeting.topic} + {canManageMeeting && } + + Status: {meeting.status} + Transcript + + {meeting.transcript || 'No transcript provided. Edit to add one.'} + + + )} + + + + + {/* AI Tools Card (Full Width) */} + + + + + AI Tools + {(canManageMeeting && !isEditingSummary) && } + + + {canManageMeeting && } + + Summary + {isEditingSummary ? ( + <> + setEditData({...editData, summary: e.target.value})} sx={{ mt: 1 }} /> + + + ) : ( + + {summaryTask && ( + + Generating... + + )} + {meeting.summary || (summaryTask ? '' : 'No summary generated yet.')} + + )} + + {canManageMeeting && ( + + + {previewedItems.length > 0 && ( + + + Context/ItemActionOwnerDue Date + {previewedItems.map((item, index) => ({item.item}{item.action}{item.owner}{item.due_date}))} +
+ +
+ )} +
+ )} +
+
+
+ + {/* Action Items List Card (Full Width) */} + + + + + Action Items + + + + + ContextActionOwnerDue DateStatusAttachmentActions + + {actionItems.map((item) => { + const isEditing = editingActionItemId === item.id; + const canEditItem = currentUser && (currentUser.role === 'admin' || String(currentUser.id) === String(item.owner_id)); + return ( + + {isEditing ? setEditActionItemData({...editActionItemData, item: e.target.value})} fullWidth /> : item.item} + {isEditing ? setEditActionItemData({...editActionItemData, action: e.target.value})} fullWidth /> : item.action} + {isEditing ? : item.owner_name} + {isEditing ? setEditActionItemData({...editActionItemData, due_date: e.target.value})} InputLabelProps={{ shrink: true }} fullWidth /> : item.due_date} + {isEditing ? : item.status} + + {isEditing ? : (item.attachment_path && downloadFile(item.attachment_path)}>)} + {isEditing && attachmentFile && {attachmentFile.name}} + + + {isEditing ? handleSaveActionItemClick(item.id)}> : {canEditItem && handleEditActionItemClick(item)}>}{canManageMeeting && handleDeleteActionItemClick(item.id)}>}} + + + ); + })} + +
+
+
+
+
+
+ + setIsAddActionItemOpen(false)} fullWidth maxWidth="sm"> + Add New Action Item + + setNewActionItem({...newActionItem, item: e.target.value})} /> + setNewActionItem({...newActionItem, action: e.target.value})} /> + Owner + setNewActionItem({...newActionItem, due_date: e.target.value})} /> + + {attachmentFile && {attachmentFile.name}} + + + +
+ ); +}; + +export default MeetingDetailPage; \ No newline at end of file diff --git a/frontend/src/pages/ProcessingPage.jsx b/frontend/src/pages/ProcessingPage.jsx new file mode 100644 index 0000000..1050091 --- /dev/null +++ b/frontend/src/pages/ProcessingPage.jsx @@ -0,0 +1,228 @@ +import React, { useState, useEffect, useCallback } from 'react'; +import { + Box, Typography, Paper, Button, CircularProgress, Alert, Grid, Card, CardContent, Chip, LinearProgress, TextField, Select, MenuItem, FormControl, InputLabel, IconButton, Tooltip +} from '@mui/material'; +import ContentCopyIcon from '@mui/icons-material/ContentCopy'; +import DownloadIcon from '@mui/icons-material/Download'; +import { + extractAudio, + transcribeAudio, + translateText, + pollTaskStatus, + stopTask, + downloadFile +} from '../services/api'; + +const TaskMonitor = ({ task, onStop, title, children }) => { + if (!task) return null; + const colorMap = { PENDING: 'default', PROGRESS: 'info', SUCCESS: 'success', FAILURE: 'error', REVOKED: 'warning' }; + const progress = task.info?.total ? (task.info.current / task.info.total * 100) : null; + const isRunning = task.state === 'PENDING' || task.state === 'PROGRESS'; + + return ( + + + {title} + {task.state && } + + {task.info?.status_msg && {task.info.status_msg}} + {isRunning && !progress && } + {progress && } + {task.state === 'FAILURE' && {task.info?.error || 'Task failed.'}} + {isRunning && + } + {children} + + ); +}; + +const ProcessingPage = () => { + const [tasks, setTasks] = useState({}); + const [error, setError] = useState(''); + const [copySuccess, setCopySuccess] = useState(''); + + const [extractFile, setExtractFile] = useState(null); + const [transcribeFile, setTranscribeFile] = useState(null); + const [transcribedText, setTranscribedText] = useState(''); + const [translateFile, setTranslateFile] = useState(null); + const [translateTextContent, setTranslateTextContent] = useState(''); + const [translateLang, setTranslateLang] = useState('繁體中文'); + const [customLang, setCustomLang] = useState(''); + const [translatedResult, setTranslatedResult] = useState(''); + + const handleCopyToClipboard = (text, type) => { + navigator.clipboard.writeText(text).then(() => { + setCopySuccess(type); + setTimeout(() => setCopySuccess(''), 2000); + }); + }; + + const handleTranslateFileUpload = (e) => { + const file = e.target.files[0]; + if (file) { + setTranslateFile(file); + const reader = new FileReader(); + reader.onload = (evt) => setTranslateTextContent(evt.target.result); + reader.readAsText(file); + } + }; + + // This function now ONLY updates the main tasks object. + const handleTaskUpdate = useCallback((key, updatedTask) => { + setTasks(prev => ({ ...prev, [key]: updatedTask })); + }, []); + + // This new useEffect handles the side-effects of a task completing. + useEffect(() => { + const transcribeTask = tasks.transcribe; + if (transcribeTask?.state === 'SUCCESS' && transcribeTask.info?.content) { + setTranscribedText(transcribeTask.info.content); + } + + const translateTask = tasks.translate; + if (translateTask?.state === 'SUCCESS' && translateTask.info?.content) { + setTranslatedResult(translateTask.info.content); + } + }, [tasks]); + + + useEffect(() => { + const intervalIds = Object.entries(tasks).map(([key, task]) => { + if (task && (task.state === 'PENDING' || task.state === 'PROGRESS')) { + const intervalId = setInterval(async () => { + try { + const updatedTask = await pollTaskStatus(task.status_url); + // Pass the full task object to avoid stale closures + handleTaskUpdate(key, { ...task, ...updatedTask }); + if (['SUCCESS', 'FAILURE', 'REVOKED'].includes(updatedTask.state)) { + clearInterval(intervalId); + } + } catch (err) { + handleTaskUpdate(key, { ...task, state: 'FAILURE', info: { ...task.info, error: 'Polling failed.' } }); + clearInterval(intervalId); + } + }, 2000); + return intervalId; + } + return null; + }).filter(Boolean); + + return () => intervalIds.forEach(clearInterval); + }, [tasks, handleTaskUpdate]); + + const handleStartTask = async (key, taskFn, ...args) => { + setError(''); + setTasks(prev => ({ ...prev, [key]: { state: 'PENDING', info: { status_msg: 'Initializing...' } } })); + try { + const result = await taskFn(...args); + setTasks(prev => ({ ...prev, [key]: { ...prev[key], task_id: result.task_id, status_url: result.status_url, state: 'PENDING' } })); + } catch (err) { + const errorMsg = err.response?.data?.error || `Failed to start ${key} task.`; + setError(errorMsg); + setTasks(prev => ({ ...prev, [key]: { state: 'FAILURE', info: { error: errorMsg } } })); + } + }; + + const handleStopTask = async (taskId) => { + if (!taskId) return; + try { + await stopTask(taskId); + const taskKey = Object.keys(tasks).find(k => tasks[k].task_id === taskId); + if (taskKey) setTasks(prev => ({ ...prev, [taskKey]: { ...prev[taskKey], state: 'REVOKED' } })); + } catch (err) { + setError('Failed to stop the task.'); + } + }; + + return ( + + Processing Tools + {error && {error}} + + {/* Left Column: Extract Audio */} + + + + 1. Extract Audio + Extract audio track from a video file. + + {extractFile && {extractFile.name}} + + + {tasks.extract?.state === 'SUCCESS' && tasks.extract.info.download_filename && + + } + + + + + + {/* Right Column: Transcribe and Translate */} + + + {/* Top-Right: Transcribe */} + + + + 2. Transcribe Audio to Text + + {transcribeFile && {transcribeFile.name}} + + + + {(tasks.transcribe?.state === 'PENDING' || tasks.transcribe?.state === 'PROGRESS') && } + {transcribedText || 'Transcription will appear here.'} + {transcribedText && handleCopyToClipboard(transcribedText, 'transcribed')} sx={{position: 'absolute', top: 5, right: 5}}>} + + {tasks.transcribe?.state === 'SUCCESS' && tasks.transcribe.info.result_path && } + + + + {/* Bottom-Right: Translate */} + + + + 3. Translate Text + setTranslateTextContent(e.target.value)} sx={{ mt: 2 }} /> + + + {translateFile && {translateFile.name}} + + + Target Language + + + {translateLang === 'Other' && setCustomLang(e.target.value)} sx={{ mt: { xs: 2, sm: 0 } }} />} + + + + + {(tasks.translate?.state === 'PENDING' || tasks.translate?.state === 'PROGRESS') && } + {translatedResult || 'Translation will appear here.'} + {translatedResult && handleCopyToClipboard(translatedResult, 'translated')} sx={{position: 'absolute', top: 5, right: 5}}>} + + {tasks.translate?.state === 'SUCCESS' && tasks.translate.info.result_path && } + + + + + + + + ); +}; + +export default ProcessingPage; diff --git a/frontend/src/services/api.js b/frontend/src/services/api.js new file mode 100644 index 0000000..cfee167 --- /dev/null +++ b/frontend/src/services/api.js @@ -0,0 +1,94 @@ +import axios from 'axios'; + +const api = axios.create({ + baseURL: '/api', + withCredentials: false, +}); + +api.interceptors.request.use((config) => { + const token = localStorage.getItem('token'); + if (token) { + config.headers.Authorization = `Bearer ${token}`; + } + return config; +}); + +const unwrap = (promise) => promise.then((r) => r.data); + +// --- Task Management --- +export const pollTaskStatus = (statusUrl) => unwrap(api.get(statusUrl)); +export const stopTask = (taskId) => unwrap(api.post(`/task/${taskId}/stop`)); +export const downloadFile = async (filename) => { + const res = await api.get(`/download/${filename}`, { responseType: 'blob' }); + const blob = new Blob([res.data], { type: res.headers['content-type'] }); + const link = document.createElement('a'); + link.href = window.URL.createObjectURL(blob); + link.setAttribute('download', filename); + document.body.appendChild(link); + link.click(); + document.body.removeChild(link); + window.URL.revokeObjectURL(link.href); +}; + + +// --- Authentication --- +export const login = (username, password) => + unwrap(api.post('/login', { username, password })); + +export const register = (username, password) => + unwrap(api.post('/register', { username, password })); + +// --- Admin --- +export const getUsers = () => unwrap(api.get('/admin/users')); // For Admin Page +export const getAllUsers = () => unwrap(api.get('/users')); // For dropdowns +export const createUser = (userData) => unwrap(api.post('/admin/users', userData)); +export const deleteUser = (userId) => unwrap(api.delete(`/admin/users/${userId}`)); +export const changeUserPassword = (userId, password) => + unwrap(api.put(`/admin/users/${userId}/password`, { password })); + +// --- Meetings --- +export const getMeetings = () => unwrap(api.get('/meetings')); +export const createMeeting = (topic, meetingDate) => unwrap(api.post('/meetings', { topic, meeting_date: meetingDate })); +export const getMeetingDetails = (meetingId) => unwrap(api.get(`/meetings/${meetingId}`)); +export const updateMeeting = (meetingId, data) => unwrap(api.put(`/meetings/${meetingId}`, data)); +export const deleteMeeting = (meetingId) => unwrap(api.delete(`/meetings/${meetingId}`)); +export const summarizeMeeting = (meetingId) => unwrap(api.post(`/meetings/${meetingId}/summarize`)); + +// --- Independent Tools --- +const startFileUploadTask = async (endpoint, file, options = {}) => { + const formData = new FormData(); + formData.append('file', file); + for (const key in options) { + formData.append(key, options[key]); + } + return unwrap(api.post(endpoint, formData, { + headers: { 'Content-Type': 'multipart/form-data' }, + })); +}; + +export const extractAudio = (file) => + startFileUploadTask('/tools/extract_audio', file); + +export const transcribeAudio = (file) => + startFileUploadTask('/tools/transcribe_audio', file); + +export const translateText = (text, target_language) => + unwrap(api.post('/tools/translate_text', { text, target_language })); + +// --- AI Previews (for Meeting Page) --- +export const previewActionItems = (text) => + unwrap(api.post('/action-items/preview', { text })); + +// --- Action Items --- +export const getActionItemsForMeeting = (meetingId) => unwrap(api.get(`/meetings/${meetingId}/action_items`)); +export const createActionItem = (payload) => unwrap(api.post('/action-items', payload)); +export const batchSaveActionItems = (meetingId, items) => unwrap(api.post(`/meetings/${meetingId}/action-items/batch`, { items })); +export const updateActionItem = (itemId, updateData) => unwrap(api.put(`/action_items/${itemId}`, updateData)); +export const deleteActionItem = (itemId) => unwrap(api.delete(`/action_items/${itemId}`)); +export const uploadActionItemAttachment = (itemId, file) => { + const formData = new FormData(); + formData.append('file', file); + return unwrap(api.post(`/action_items/${itemId}/upload`, formData, { + headers: { 'Content-Type': 'multipart/form-data' }, + })); +}; diff --git a/frontend/vite.config.js b/frontend/vite.config.js new file mode 100644 index 0000000..d8dc48f --- /dev/null +++ b/frontend/vite.config.js @@ -0,0 +1,15 @@ +import { defineConfig } from 'vite' +import react from '@vitejs/plugin-react' + +// https://vitejs.dev/config/ +export default defineConfig({ + plugins: [react()], + server: { + proxy: { + '/api': { + target: 'http://127.0.0.1:12000', + changeOrigin: true, + }, + }, + }, +}) diff --git a/migrations/README b/migrations/README new file mode 100644 index 0000000..0e04844 --- /dev/null +++ b/migrations/README @@ -0,0 +1 @@ +Single-database configuration for Flask. diff --git a/migrations/__pycache__/env.cpython-312.pyc b/migrations/__pycache__/env.cpython-312.pyc new file mode 100644 index 0000000..5f89e0a Binary files /dev/null and b/migrations/__pycache__/env.cpython-312.pyc differ diff --git a/migrations/alembic.ini b/migrations/alembic.ini new file mode 100644 index 0000000..ec9d45c --- /dev/null +++ b/migrations/alembic.ini @@ -0,0 +1,50 @@ +# A generic, single database configuration. + +[alembic] +# template used to generate migration files +# file_template = %%(rev)s_%%(slug)s + +# set to 'true' to run the environment during +# the 'revision' command, regardless of autogenerate +# revision_environment = false + + +# Logging configuration +[loggers] +keys = root,sqlalchemy,alembic,flask_migrate + +[handlers] +keys = console + +[formatters] +keys = generic + +[logger_root] +level = WARN +handlers = console +qualname = + +[logger_sqlalchemy] +level = WARN +handlers = +qualname = sqlalchemy.engine + +[logger_alembic] +level = INFO +handlers = +qualname = alembic + +[logger_flask_migrate] +level = INFO +handlers = +qualname = flask_migrate + +[handler_console] +class = StreamHandler +args = (sys.stderr,) +level = NOTSET +formatter = generic + +[formatter_generic] +format = %(levelname)-5.5s [%(name)s] %(message)s +datefmt = %H:%M:%S diff --git a/migrations/env.py b/migrations/env.py new file mode 100644 index 0000000..20e5fae --- /dev/null +++ b/migrations/env.py @@ -0,0 +1,104 @@ +import logging +from logging.config import fileConfig + +from flask import current_app + +from alembic import context + +# this is the Alembic Config object, which provides +# access to the values within the .ini file in use. +config = context.config + +# Interpret the config file for Python logging. +# This line sets up loggers basically. +fileConfig(config.config_file_name) +logger = logging.getLogger('alembic.env') + + +def get_engine(): + try: + # this works with Flask-SQLAlchemy<3 and Alchemical + return current_app.extensions['migrate'].db.get_engine() + except (TypeError, AttributeError): + # this works with Flask-SQLAlchemy>=3 + return current_app.extensions['migrate'].db.engine + + +def get_engine_url(): + try: + return get_engine().url.render_as_string(hide_password=False).replace( + '%', '%%') + except AttributeError: + return str(get_engine().url).replace('%', '%%') + + +# add your model's MetaData object here +# for 'autogenerate' support +# from myapp import mymodel +# target_metadata = mymodel.Base.metadata +config.set_main_option('sqlalchemy.url', get_engine_url()) +target_db = current_app.extensions['migrate'].db + +# other values from the config, defined by the needs of env.py, +# can be acquired: +# my_important_option = config.get_main_option("my_important_option") +# ... etc. + + +def get_metadata(): + if hasattr(target_db, 'metadatas'): + return target_db.metadatas[None] + return target_db.metadata + + +def run_migrations_offline(): + """Run migrations in 'offline' mode.""" + url = config.get_main_option("sqlalchemy.url") + context.configure( + url=url, target_metadata=get_metadata(), literal_binds=True + ) + + with context.begin_transaction(): + context.run_migrations() + +# --- NEW --- Function to filter tables by prefix +def include_name(name, type_, parent_names): + if type_ == 'table': + return name.startswith('ms_') + else: + return True +# --- END NEW --- + +def run_migrations_online(): + """Run migrations in 'online' mode.""" + + def process_revision_directives(context, revision, directives): + if getattr(config.cmd_opts, 'autogenerate', False): + script = directives[0] + if script.upgrade_ops.is_empty(): + directives[:] = [] + logger.info('No changes in schema detected.') + + conf_args = current_app.extensions['migrate'].configure_args + if conf_args.get("process_revision_directives") is None: + conf_args["process_revision_directives"] = process_revision_directives + + connectable = get_engine() + + with connectable.connect() as connection: + context.configure( + connection=connection, + target_metadata=get_metadata(), + # --- MODIFIED --- Add the include_name filter + include_name=include_name, + **conf_args + ) + + with context.begin_transaction(): + context.run_migrations() + + +if context.is_offline_mode(): + run_migrations_offline() +else: + run_migrations_online() diff --git a/migrations/script.py.mako b/migrations/script.py.mako new file mode 100644 index 0000000..2c01563 --- /dev/null +++ b/migrations/script.py.mako @@ -0,0 +1,24 @@ +"""${message} + +Revision ID: ${up_revision} +Revises: ${down_revision | comma,n} +Create Date: ${create_date} + +""" +from alembic import op +import sqlalchemy as sa +${imports if imports else ""} + +# revision identifiers, used by Alembic. +revision = ${repr(up_revision)} +down_revision = ${repr(down_revision)} +branch_labels = ${repr(branch_labels)} +depends_on = ${repr(depends_on)} + + +def upgrade(): + ${upgrades if upgrades else "pass"} + + +def downgrade(): + ${downgrades if downgrades else "pass"} diff --git a/migrations/versions/919aff0aa44b_implement_user_centric_status_and_.py b/migrations/versions/919aff0aa44b_implement_user_centric_status_and_.py new file mode 100644 index 0000000..229ac50 --- /dev/null +++ b/migrations/versions/919aff0aa44b_implement_user_centric_status_and_.py @@ -0,0 +1,34 @@ +"""Implement user-centric status and transcript field for meetings + +Revision ID: 919aff0aa44b +Revises: ac069534da31 +Create Date: 2025-08-15 09:20:02.369230 + +""" +from alembic import op +import sqlalchemy as sa +from sqlalchemy.dialects import mysql + +# revision identifiers, used by Alembic. +revision = '919aff0aa44b' +down_revision = 'ac069534da31' +branch_labels = None +depends_on = None + + +def upgrade(): + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table('ms_meetings', schema=None) as batch_op: + batch_op.add_column(sa.Column('transcript', sa.Text(), nullable=True)) + batch_op.drop_column('transcript_file_path') + + # ### end Alembic commands ### + + +def downgrade(): + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table('ms_meetings', schema=None) as batch_op: + batch_op.add_column(sa.Column('transcript_file_path', mysql.VARCHAR(length=512), nullable=True)) + batch_op.drop_column('transcript') + + # ### end Alembic commands ### diff --git a/migrations/versions/__pycache__/3b11caf37983_initial_migration_with_users_meetings_.cpython-312.pyc b/migrations/versions/__pycache__/3b11caf37983_initial_migration_with_users_meetings_.cpython-312.pyc new file mode 100644 index 0000000..6c7dd22 Binary files /dev/null and b/migrations/versions/__pycache__/3b11caf37983_initial_migration_with_users_meetings_.cpython-312.pyc differ diff --git a/migrations/versions/__pycache__/919aff0aa44b_implement_user_centric_status_and_.cpython-312.pyc b/migrations/versions/__pycache__/919aff0aa44b_implement_user_centric_status_and_.cpython-312.pyc new file mode 100644 index 0000000..895ddcc Binary files /dev/null and b/migrations/versions/__pycache__/919aff0aa44b_implement_user_centric_status_and_.cpython-312.pyc differ diff --git a/migrations/versions/__pycache__/ac069534da31_add_status_and_result_fields_to_meeting.cpython-312.pyc b/migrations/versions/__pycache__/ac069534da31_add_status_and_result_fields_to_meeting.cpython-312.pyc new file mode 100644 index 0000000..3154008 Binary files /dev/null and b/migrations/versions/__pycache__/ac069534da31_add_status_and_result_fields_to_meeting.cpython-312.pyc differ diff --git a/migrations/versions/ac069534da31_add_status_and_result_fields_to_meeting.py b/migrations/versions/ac069534da31_add_status_and_result_fields_to_meeting.py new file mode 100644 index 0000000..78b44ec --- /dev/null +++ b/migrations/versions/ac069534da31_add_status_and_result_fields_to_meeting.py @@ -0,0 +1,52 @@ +"""Add status and result fields to Meeting + +Revision ID: ac069534da31 +Revises: +Create Date: 2025-08-15 08:38:19.572077 + +""" +from alembic import op +import sqlalchemy as sa +from sqlalchemy.dialects import mysql + +# revision identifiers, used by Alembic. +revision = 'ac069534da31' +down_revision = None +branch_labels = None +depends_on = None + + +def upgrade(): + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table('ms_action_items', schema=None) as batch_op: + batch_op.alter_column('owner_id', + existing_type=mysql.INTEGER(), + nullable=True) + + with op.batch_alter_table('ms_meetings', schema=None) as batch_op: + batch_op.add_column(sa.Column('status', sa.String(length=50), nullable=False)) + batch_op.add_column(sa.Column('transcript_file_path', sa.String(length=512), nullable=True)) + batch_op.add_column(sa.Column('summary', sa.Text(), nullable=True)) + batch_op.alter_column('created_by_id', + existing_type=mysql.INTEGER(), + nullable=True) + + # ### end Alembic commands ### + + +def downgrade(): + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table('ms_meetings', schema=None) as batch_op: + batch_op.alter_column('created_by_id', + existing_type=mysql.INTEGER(), + nullable=False) + batch_op.drop_column('summary') + batch_op.drop_column('transcript_file_path') + batch_op.drop_column('status') + + with op.batch_alter_table('ms_action_items', schema=None) as batch_op: + batch_op.alter_column('owner_id', + existing_type=mysql.INTEGER(), + nullable=False) + + # ### end Alembic commands ### diff --git a/models.py b/models.py new file mode 100644 index 0000000..9f94829 --- /dev/null +++ b/models.py @@ -0,0 +1,93 @@ +from flask_sqlalchemy import SQLAlchemy +from sqlalchemy.sql import func +from flask_bcrypt import Bcrypt + +db = SQLAlchemy() +bcrypt = Bcrypt() + +class User(db.Model): + __tablename__ = 'ms_users' + id = db.Column(db.Integer, primary_key=True) + username = db.Column(db.String(80), unique=True, nullable=False) + password_hash = db.Column(db.String(128), nullable=False) + role = db.Column(db.String(20), nullable=False, default='user') # 'user' or 'admin' + created_at = db.Column(db.DateTime(timezone=True), server_default=func.now()) + + def set_password(self, password): + self.password_hash = bcrypt.generate_password_hash(password).decode('utf-8') + + def check_password(self, password): + return bcrypt.check_password_hash(self.password_hash, password) + + def to_dict(self): + return { + 'id': self.id, + 'username': self.username, + 'role': self.role, + 'created_at': self.created_at.isoformat() if self.created_at else None + } + +class Meeting(db.Model): + __tablename__ = 'ms_meetings' + id = db.Column(db.Integer, primary_key=True) + topic = db.Column(db.String(255), nullable=False) + meeting_date = db.Column(db.DateTime(timezone=True), nullable=False) + created_by_id = db.Column(db.Integer, db.ForeignKey('ms_users.id'), nullable=True) + created_at = db.Column(db.DateTime(timezone=True), server_default=func.now()) + + # New user-centric status + status = db.Column(db.String(50), nullable=False, default='To Do') # 'To Do', 'In Progress', 'Completed' + + # Field to store the pasted transcript + transcript = db.Column(db.Text, nullable=True) + + # AI-generated summary + summary = db.Column(db.Text, nullable=True) + + # Removed transcript_file_path as it's no longer needed in this workflow + + creator = db.relationship('User', backref=db.backref('meetings', lazy=True)) + action_items = db.relationship('ActionItem', backref='meeting', lazy='dynamic', cascade="all, delete-orphan") + + def to_dict(self): + return { + 'id': self.id, + 'topic': self.topic, + 'meeting_date': self.meeting_date.isoformat() if self.meeting_date else None, + 'created_by_id': self.created_by_id, + 'owner_name': self.creator.username if self.creator else None, + 'created_at': f"{self.created_at.isoformat()}Z" if self.created_at else None, + 'action_item_count': self.action_items.count(), + 'status': self.status, + 'transcript': self.transcript, + 'summary': self.summary, + } + +class ActionItem(db.Model): + __tablename__ = 'ms_action_items' + id = db.Column(db.Integer, primary_key=True) + meeting_id = db.Column(db.Integer, db.ForeignKey('ms_meetings.id'), nullable=False) + item = db.Column(db.Text, nullable=True) + action = db.Column(db.Text, nullable=False) + owner_id = db.Column(db.Integer, db.ForeignKey('ms_users.id'), nullable=True) + due_date = db.Column(db.Date, nullable=True) + status = db.Column(db.String(50), nullable=False, default='pending') # e.g., 'pending', 'in_progress', 'completed' + attachment_path = db.Column(db.String(255), nullable=True) + created_at = db.Column(db.DateTime(timezone=True), server_default=func.now()) + + owner = db.relationship('User', backref=db.backref('action_items', lazy=True)) + + def to_dict(self): + return { + 'id': self.id, + 'meeting_id': self.meeting_id, + 'meeting_topic': self.meeting.topic if self.meeting else None, + 'item': self.item, + 'action': self.action, + 'owner_id': self.owner_id, + 'owner_name': self.owner.username if self.owner else None, + 'due_date': self.due_date.isoformat() if self.due_date else None, + 'status': self.status, + 'attachment_path': self.attachment_path, + 'created_at': self.created_at.isoformat() if self.created_at else None + } diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..5ff6d25 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,28 @@ +Flask==2.2.5 +celery==5.3.6 +redis==4.5.4 +# For NVIDIA GPU (CUDA 11.8) support, use these lines: +torch --extra-index-url https://download.pytorch.org/whl/cu118 +torchaudio --extra-index-url https://download.pytorch.org/whl/cu118 +# For CPU-only, comment out the two lines above and uncomment the two lines below: +# torch +# torchaudio +openai-whisper +moviepy +opencc-python-reimplemented +ffmpeg-python +python-dotenv +gunicorn +demucs +soundfile +gevent # Added for celery on windows + +# New dependencies for User Management and Database +Flask-SQLAlchemy +Flask-Migrate +PyMySQL +Flask-JWT-Extended +Flask-Bcrypt + +# Dependency for calling external APIs +requests diff --git a/services/__pycache__/dify_client.cpython-312.pyc b/services/__pycache__/dify_client.cpython-312.pyc new file mode 100644 index 0000000..39614f1 Binary files /dev/null and b/services/__pycache__/dify_client.cpython-312.pyc differ diff --git a/services/dify_client.py b/services/dify_client.py new file mode 100644 index 0000000..d2f80c1 --- /dev/null +++ b/services/dify_client.py @@ -0,0 +1,93 @@ +# services/dify_client.py +import os, json, re, requests +from dotenv import load_dotenv + +load_dotenv() + +DIFY_BASE = os.getenv("DIFY_API_BASE_URL", "https://api.dify.ai/v1") +TIMEOUT = 60 + +def _post_request(endpoint: str, api_key: str, payload: dict): + """Generic function to post a request to a Dify endpoint.""" + if not api_key: + raise RuntimeError("Dify API key is not set") + + url = f"{DIFY_BASE}{endpoint}" + headers = { + "Authorization": f"Bearer {api_key}", + "Content-Type": "application/json", + } + + # For debugging the exact payload sent to the API + print(f"Sending Dify request to {url}: {json.dumps(payload, indent=2, ensure_ascii=False)}") + + resp = requests.post(url, headers=headers, json=payload, timeout=TIMEOUT) + + if resp.status_code != 200: + print(f"Dify API Error Response: {resp.text}") + resp.raise_for_status() + + data = resp.json() + return data.get("answer") or data + +def translate_text(text: str, target_lang: str, user_id: str = "system") -> str: + """ + Calls the Dify CHAT API for translation, mimicking a user query. + """ + api_key = os.getenv("DIFY_TRANSLATOR_API_KEY") + # Combine all information into a single query string, as expected by the prompt + query = f"目標語言:{target_lang}\n需翻譯內容:\n{text}" + + payload = { + "inputs": {}, # Chatbot apps generally don't use inputs here + "response_mode": "blocking", + "user": user_id, + "query": query, + # conversation_id is optional for single-turn interactions + } + return _post_request("/chat-messages", api_key, payload) + +def summarize_text(text: str, user_id: str = "system") -> str: + api_key = os.getenv("DIFY_SUMMARIZER_API_KEY") + payload = { + "inputs": {}, + "response_mode": "blocking", + "user": user_id, + "query": text, + } + return _post_request("/chat-messages", api_key, payload) + +def extract_action_items(text: str, user_id: str = "system") -> list[dict]: + api_key = os.getenv("DIFY_ACTION_EXTRACTOR_API_KEY") + payload = { + "inputs": {}, + "response_mode": "blocking", + "user": user_id, + "query": text, + } + raw = _post_request("/chat-messages", api_key, payload) + + # Fault tolerance for JSON parsing + s = str(raw).strip() + s = re.sub(r"^```(?:json)?|```$", "", s, flags=re.IGNORECASE|re.MULTILINE).strip() + if not (s.startswith("[") and s.endswith("]")): + m = re.search(r"[\s\S]*\[[\s\S]*\][\s\S]*", s) + if m: s = m.group(0) + + items = json.loads(s) + if not isinstance(items, list): + raise ValueError("Extractor did not return a list") + + # Normalize keys for database storage + normalized = [] + for i in items: + for k in ("item", "action", "owner", "duedate"): + if k not in i: + raise ValueError(f"Extractor item is missing required key: {k}") + normalized.append({ + "item": i["item"], + "action": i["action"], + "owner": i["owner"], + "due_date": i["duedate"], + }) + return normalized \ No newline at end of file diff --git a/static/css/style.css b/static/css/style.css new file mode 100644 index 0000000..9971d18 --- /dev/null +++ b/static/css/style.css @@ -0,0 +1,38 @@ +/* static/css/style.css */ +body { + background-color: #f8f9fa; +} + +.container { + max-width: 960px; +} + +.card-header-tabs { + margin-bottom: -1px; +} + +.nav-link { + color: #6c757d; +} + +.nav-link.active { + color: #000; + background-color: #fff; + border-color: #dee2e6 #dee2e6 #fff; +} + +.result-preview { + white-space: pre-wrap; + word-wrap: break-word; + max-height: 400px; + overflow-y: auto; + font-family: 'Courier New', Courier, monospace; +} + +.action-btn:disabled { + cursor: not-allowed; +} + +.progress-bar { + transition: width 0.6s ease; +} diff --git a/static/js/script.js b/static/js/script.js new file mode 100644 index 0000000..b00f9b0 --- /dev/null +++ b/static/js/script.js @@ -0,0 +1,275 @@ +document.addEventListener('DOMContentLoaded', function() { + // --- Global variables --- + let statusInterval; + let currentTaskType = ''; + let summaryConversationId = null; + let lastSummaryText = ''; + + // --- DOM Elements --- + const progressContainer = document.getElementById('progress-container'); + const statusText = document.getElementById('status-text'); + const progressBar = document.getElementById('progress-bar'); + const resultContainer = document.getElementById('result-container'); + const textResultPreview = document.getElementById('text-result-preview'); + const downloadLink = document.getElementById('download-link'); + const revisionArea = document.getElementById('revision-area'); + const allActionButtons = document.querySelectorAll('.action-btn'); + + // --- Tab Switching Logic --- + const tabButtons = document.querySelectorAll('#myTab button'); + tabButtons.forEach(button => { + button.addEventListener('shown.bs.tab', function() { + resetUiForNewTask(); + }); + }); + + // --- Event Listeners for all action buttons --- + allActionButtons.forEach(button => { + button.addEventListener('click', handleActionClick); + }); + + function handleActionClick(event) { + const button = event.currentTarget; + currentTaskType = button.dataset.task; + + resetUiForNewTask(); + button.disabled = true; + button.innerHTML = ' 處理中...'; + progressContainer.style.display = 'block'; + + if (currentTaskType === 'summarize_text') { + const fileInput = document.getElementById('summary-file-input'); + const file = fileInput.files[0]; + + if (file) { + const reader = new FileReader(); + reader.onload = function(e) { + const fileContent = e.target.result; + startSummarizeTask(fileContent); + }; + reader.onerror = function() { + handleError("讀取檔案時發生錯誤。"); + }; + reader.readAsText(file); + } else { + const textContent = document.getElementById('summary-source-text').value; + if (!textContent.trim()) { + alert('請貼上文字或選擇檔案!'); + resetButtons(); + return; + } + startSummarizeTask(textContent); + } + return; + } + + let endpoint = ''; + let formData = new FormData(); + let body = null; + let fileInput; + + switch (currentTaskType) { + case 'extract_audio': + endpoint = '/extract_audio'; + fileInput = document.getElementById('video-file'); + break; + + case 'transcribe_audio': + endpoint = '/transcribe_audio'; + fileInput = document.getElementById('audio-file'); + formData.append('language', document.getElementById('lang-select').value); + if (document.getElementById('use-demucs').checked) { + formData.append('use_demucs', 'on'); + } + break; + + case 'translate_text': + endpoint = '/translate_text'; + fileInput = document.getElementById('transcript-file'); + formData.append('target_language', document.getElementById('translate-lang-select').value); + break; + + case 'revise_summary': + endpoint = '/summarize_text'; + const instruction = document.getElementById('revision-instruction').value; + if (!lastSummaryText) { alert('請先生成初版結論!'); resetButtons(); return; } + if (!instruction.trim()) { alert('請輸入修改指示!'); resetButtons(); return; } + body = JSON.stringify({ + text_content: lastSummaryText, + revision_instruction: instruction, + target_language: document.getElementById('summary-lang-select').value, + conversation_id: summaryConversationId + }); + startFetchTask(endpoint, body, { 'Content-Type': 'application/json' }); + return; + + default: + console.error('Unknown task type:', currentTaskType); + resetButtons(); + return; + } + + if (!fileInput || !fileInput.files[0]) { + alert('請選擇一個檔案!'); + resetButtons(); + return; + } + formData.append('file', fileInput.files[0]); + body = formData; + + startFetchTask(endpoint, body); + } + + function startSummarizeTask(textContent) { + summaryConversationId = null; + lastSummaryText = textContent; + const body = JSON.stringify({ + text_content: textContent, + target_language: document.getElementById('summary-lang-select').value + }); + startFetchTask('/summarize_text', body, { 'Content-Type': 'application/json' }); + } + + function startFetchTask(endpoint, body, headers = {}) { + updateProgress(0, '準備上傳與處理...'); + fetch(endpoint, { + method: 'POST', + body: body, + headers: headers + }) + .then(response => { + if (!response.ok) { + return response.json().then(err => { throw new Error(err.error || '伺服器錯誤') }); + } + return response.json(); + }) + .then(data => { + if (data.task_id) { + statusInterval = setInterval(() => checkTaskStatus(data.status_url), 2000); + } else { + handleError(data.error || '未能啟動背景任務'); + } + }) + .catch(error => { + handleError(error.message || '請求失敗'); + }); + } + + function checkTaskStatus(statusUrl) { + fetch(statusUrl) + .then(response => response.json()) + .then(data => { + const info = data.info || {}; + if (data.state === 'PROGRESS') { + updateProgress(info.current, info.status, info.total); + const previewContent = info.content || info.summary || info.preview; + if (previewContent) { + resultContainer.style.display = 'block'; + textResultPreview.textContent = previewContent; + textResultPreview.style.display = 'block'; + } + } else if (data.state === 'SUCCESS') { + clearInterval(statusInterval); + updateProgress(100, info.status || '完成!', 100); + displayResult(info); + resetButtons(); + } else if (data.state === 'FAILURE') { + clearInterval(statusInterval); + handleError(info.exc_message || '任務執行失敗'); + } + }) + .catch(error => { + clearInterval(statusInterval); + handleError('查詢進度時發生網路錯誤: ' + error); + }); + } + + function updateProgress(current, text, total = 100) { + const percent = total > 0 ? Math.round((current / total) * 100) : 0; + progressBar.style.width = percent + '%'; + progressBar.setAttribute('aria-valuenow', percent); + progressBar.textContent = percent + '%'; + statusText.textContent = text; + } + + function displayResult(info) { + resultContainer.style.display = 'block'; + + const content = info.content || info.summary; + if (content) { + textResultPreview.textContent = content; + textResultPreview.style.display = 'block'; + lastSummaryText = content; + } else { + textResultPreview.style.display = 'none'; + } + + if (info.download_url) { + downloadLink.href = info.download_url; + downloadLink.style.display = 'inline-block'; + } + + if (currentTaskType === 'summarize_text' || currentTaskType === 'revise_summary') { + revisionArea.style.display = 'block'; + summaryConversationId = info.conversation_id; + } + } + + function handleError(message) { + statusText.textContent = `錯誤:${message}`; + progressBar.classList.add('bg-danger'); + resetButtons(); + } + + function resetUiForNewTask() { + if (statusInterval) clearInterval(statusInterval); + + progressContainer.style.display = 'none'; + resultContainer.style.display = 'none'; + textResultPreview.style.display = 'none'; + textResultPreview.textContent = ''; + downloadLink.style.display = 'none'; + revisionArea.style.display = 'none'; + + progressBar.style.width = '0%'; + progressBar.setAttribute('aria-valuenow', 0); + progressBar.textContent = '0%'; + progressBar.classList.remove('bg-danger'); + statusText.textContent = ''; + + resetButtons(); + } + + function resetButtons() { + allActionButtons.forEach(button => { + button.disabled = false; + const task = button.dataset.task; + let iconHtml = ''; + let text = ''; + + switch(task) { + case 'extract_audio': + iconHtml = ''; + text = '開始轉換'; + break; + case 'transcribe_audio': + iconHtml = ''; + text = '開始轉錄'; + break; + case 'translate_text': + iconHtml = ''; + text = '開始翻譯'; + break; + case 'summarize_text': + iconHtml = ''; + text = '產生初版結論'; + break; + case 'revise_summary': + iconHtml = ''; + text = '根據指示產生修改版'; + break; + } + button.innerHTML = iconHtml + text; + }); + } +}); diff --git a/tasks.py b/tasks.py new file mode 100644 index 0000000..a806ecd --- /dev/null +++ b/tasks.py @@ -0,0 +1,431 @@ +import os +import uuid +import requests +import json +import re +import io +from datetime import timedelta +from pydub import AudioSegment +from pydub.silence import split_on_silence +from moviepy import VideoFileClip +from celery_app import celery +from models import db, Meeting +import math + +from celery import Task + +class ProgressTask(Task): + def update_progress(self, current, total, status_msg, extra_info=None): + meta = {'current': current, 'total': total, 'status_msg': status_msg} + if extra_info and isinstance(extra_info, dict): + meta.update(extra_info) + self.update_state(state='PROGRESS', meta=meta) + +# --- Dify Helper Functions --- +def ask_dify(api_key: str, query: str, **kwargs): + from flask import current_app + DIFY_API_BASE_URL = current_app.config.get("DIFY_API_BASE_URL") + if not api_key or not DIFY_API_BASE_URL: + return {"answer": "Error: DIFY API settings not configured."} + + url = f"{DIFY_API_BASE_URL}/chat-messages" + headers = {"Authorization": f"Bearer {api_key}", "Content-Type": "application/json"} + payload = { + "inputs": kwargs.get("inputs", {}), + "query": query, + "user": kwargs.get("user_id", "default-tk-user"), + "response_mode": kwargs.get("response_mode", "blocking"), + "conversation_id": kwargs.get("conversation_id") + } + try: + response = requests.post(url, headers=headers, json=payload, timeout=kwargs.get("timeout_seconds", 1200)) + response.raise_for_status() + return response.json() + except requests.exceptions.RequestException as e: + return {"answer": f"Dify API request error: {e}"} + +def upload_chunk_to_dify(blob: bytes, filename="chunk.mp4"): + from flask import current_app + DIFY_API_BASE_URL = current_app.config.get("DIFY_API_BASE_URL") + API_KEY = current_app.config.get("DIFY_STT_API_KEY") + print(f"DEBUG: In upload_chunk_to_dify, using DIFY_STT_API_KEY: '{API_KEY}'") # FINAL DEBUG + + r = requests.post( + f"{DIFY_API_BASE_URL}/files/upload", + headers={"Authorization": f"Bearer {API_KEY}"}, + files={"file": (filename, io.BytesIO(blob), "audio/mp4")}, + data={"user": "ai-meeting-assistant-user"}, + timeout=300 + ) + print("Dify File Upload API Response:", r.status_code, r.text) # DEBUG PRINT + r.raise_for_status() + return r.json()["id"] + +def run_dify_stt_chat_app(file_id: str) -> str: + """ + 透過呼叫 Dify 的 chat-messages API 來執行語音轉文字。 + 適用於「進階對話型」應用。 + """ + from flask import current_app + DIFY_API_BASE_URL = current_app.config.get("DIFY_API_BASE_URL") + API_KEY = current_app.config.get("DIFY_STT_API_KEY") + + payload = { + "inputs": {}, + "query": "請將音檔轉換為文字", + "user": "ai-meeting-assistant-user", + "response_mode": "blocking", + "files": [ + { + "type": "audio", + "transfer_method": "local_file", + "upload_file_id": file_id + } + ] + } + + r = requests.post( + f"{DIFY_API_BASE_URL}/chat-messages", + headers={"Authorization": f"Bearer {API_KEY}", "Content-Type": "application/json"}, + json=payload, + timeout=600 + ) + + print("Dify STT API Response:", r.status_code, r.text) # DEBUG PRINT + r.raise_for_status() + + j = r.json() + # 在對話型應用中,答案通常在 'answer' 欄位 + return j.get("answer", "").strip() + +# --- Timestamp & Audio Chunking Helpers --- +def format_timestamp_from_ms(ms: int) -> str: + td = timedelta(milliseconds=ms) + total_seconds = td.total_seconds() + hours, remainder = divmod(total_seconds, 3600) + minutes, seconds = divmod(remainder, 60) + milliseconds = td.microseconds // 1000 + return f"{int(hours):02d}:{int(minutes):02d}:{int(seconds):02d}.{milliseconds:03d}" + +def export_audio_bytes(seg: AudioSegment) -> bytes: + buf = io.BytesIO() + seg.export(buf, format="mp4", bitrate="64k") + buf.seek(0) + return buf.read() + +def ensure_chunk_under_limits(seg: AudioSegment): + MAX_SEG_MS = 20 * 60 * 1000 # Max duration 20 minutes + MAX_BYTES = 24 * 1024 * 1024 # Dify file limit is 25MB, use 24MB as a safe buffer + + if len(export_audio_bytes(seg)) < MAX_BYTES: + return [seg] + + # If segment is too large, split by duration first + parts = [] + start_ms = 0 + while start_ms < len(seg): + end_ms = min(start_ms + MAX_SEG_MS, len(seg)) + parts.append(seg[start_ms:end_ms]) + start_ms = end_ms + + # Then, double-check byte size for each part + final_parts = [] + for p in parts: + if len(export_audio_bytes(p)) < MAX_BYTES: + final_parts.append(p) + else: + # Fallback for rare cases: split in half until compliant + sub_parts = [p] + while sub_parts: + current_part = sub_parts.pop(0) + if len(export_audio_bytes(current_part)) < MAX_BYTES: + final_parts.append(current_part) + continue + + mid_point = len(current_part) // 2 + if mid_point < 1000: # Stop splitting if it's less than a second + final_parts.append(current_part) + continue + + sub_parts.extend([current_part[:mid_point], current_part[mid_point:]]) + return final_parts + +# --- Celery Tasks --- +@celery.task(base=ProgressTask, bind=True) +def extract_audio_task(self, input_path, output_path): + try: + self.update_progress(0, 100, "Starting audio extraction...") + with VideoFileClip(input_path) as video: + video.audio.write_audiofile(output_path) + self.update_progress(100, 100, "Audio extracted successfully.") + return {'status': 'Success', 'result_path': output_path} + except Exception as e: + self.update_state(state='FAILURE', meta={'exc_type': type(e).__name__, 'exc_message': str(e)}) + +@celery.task(base=ProgressTask, bind=True) +def transcribe_audio_task(self, audio_path): + from app import app + with app.app_context(): + try: + self.update_progress(0, 100, "Loading and preparing audio file...") + audio = AudioSegment.from_file(audio_path) + + # 1. Split audio by silence + self.update_progress(5, 100, "Detecting silence to split audio into chunks...") + chunks = split_on_silence( + audio, + min_silence_len=700, + silence_thresh=-40, + keep_silence=300 + ) + if not chunks: # If no silence is detected, treat the whole audio as one chunk + chunks = [audio] + + # 2. Process chunks and ensure they are within API limits + final_segments = [] + cursor_ms = 0 + for chunk in chunks: + start_time = cursor_ms + end_time = start_time + len(chunk) + + safe_parts = ensure_chunk_under_limits(chunk) + part_start_time = start_time + for part in safe_parts: + part_end_time = part_start_time + len(part) + final_segments.append({ + "start": part_start_time, + "end": part_end_time, + "segment": part + }) + part_start_time = part_end_time + cursor_ms = end_time + + # 3. Upload chunks to Dify and get transcriptions + transcribed_lines = [] + total_segments = len(final_segments) + for i, seg_data in enumerate(final_segments): + progress = 10 + int((i / total_segments) * 85) + self.update_progress(progress, 100, f"Processing chunk {i+1} of {total_segments}...") + + audio_bytes = export_audio_bytes(seg_data["segment"]) + file_id = upload_chunk_to_dify(audio_bytes, f"chunk_{i+1}.mp4") + transcribed_text = run_dify_stt_chat_app(file_id).strip() + + if transcribed_text: + start_ts = format_timestamp_from_ms(seg_data["start"]) + end_ts = format_timestamp_from_ms(seg_data["end"]) + transcribed_lines.append(f"[{start_ts} - {end_ts}] {transcribed_text}") + + # 4. Finalize and save the result + self.update_progress(98, 100, "Finalizing transcript...") + full_content = "\n".join(transcribed_lines) + + transcript_filename = f"transcript_{uuid.uuid4()}.txt" + output_txt_path = os.path.join(app.config['UPLOAD_FOLDER'], transcript_filename) + with open(output_txt_path, "w", encoding="utf-8") as f: + f.write(full_content) + + self.update_progress(100, 100, "Transcription complete.") + return {'status': 'Success', 'content': full_content, 'result_path': transcript_filename} + + except Exception as e: + error_message = f"An error occurred: {str(e)}" + self.update_state( + state='FAILURE', + meta={'exc_type': type(e).__name__, 'exc_message': error_message} + ) + return {'status': 'Error', 'error': error_message} + +@celery.task(base=ProgressTask, bind=True) +def translate_text_task(self, text_content, target_language): + from app import app + with app.app_context(): + from services.dify_client import translate_text as dify_translate + try: + self.update_progress(0, 100, f"Starting translation to {target_language}...") + if isinstance(text_content, dict): + text_content = text_content.get('content', '') + + if not text_content or not isinstance(text_content, str): + self.update_progress(100, 100, "Translation skipped due to empty input.") + return {'status': 'Success', 'content': '', 'result_path': None, 'message': 'Input was empty.'} + + lines = text_content.strip().split('\n') + final_content = "" + + timestamp_pattern = re.compile(r'^(\s*\[\d{2}:\d{2}:\d{2}\.\d{3}\s-\s\d{2}:\d{2}:\d{2}\.\d{3}\])\s*(.*)') + + is_timestamped_input = False + if lines: + if timestamp_pattern.match(lines[0]): + is_timestamped_input = True + + if is_timestamped_input: + translated_lines = [] + total_lines = len(lines) + for i, line in enumerate(lines): + match = timestamp_pattern.match(line) + if not match: + if line.strip(): # Keep non-matching, non-empty lines + translated_lines.append(line) + continue + + timestamp = match.group(1) + original_text = match.group(2) + + # Add original line + translated_lines.append(line) + + if not original_text.strip(): + continue + + translated_text = dify_translate(text=original_text, target_lang=target_language) + # Add translated line, preserving the timestamp + translated_lines.append(f"{timestamp} {translated_text}") + + progress = int(((i + 1) / total_lines) * 100) + self.update_progress(progress, 100, f"Translating line {i+1}/{total_lines}...") + + final_content = "\n".join(translated_lines) + else: + # Handle non-timestamped text line by line for bilingual output + translated_lines = [] + total_lines = len(lines) + for i, line in enumerate(lines): + progress = int(((i + 1) / total_lines) * 98) # Leave some room for finalization + self.update_progress(progress, 100, f"Translating line {i+1}/{total_lines}...") + + original_line = line.strip() + if not original_line: + continue + + translated_text = dify_translate(text=original_line, target_lang=target_language) + translated_lines.append(original_line) + translated_lines.append(translated_text) + translated_lines.append("") # Add a blank line for readability + + final_content = "\n".join(translated_lines) + + translated_filename = f"translated_{uuid.uuid4()}.txt" + upload_folder = app.config.get('UPLOAD_FOLDER', 'uploads') + output_txt_path = os.path.join(upload_folder, translated_filename) + + os.makedirs(upload_folder, exist_ok=True) + with open(output_txt_path, "w", encoding="utf-8") as f: + f.write(final_content) + + self.update_progress(100, 100, "Translation complete.") + return {'status': 'Success', 'content': final_content, 'result_path': translated_filename} + except Exception as e: + self.update_state( + state='FAILURE', + meta={'exc_type': type(e).__name__, 'exc_message': str(e)} + ) + +@celery.task(bind=True) +def summarize_text_task(self, meeting_id): + from app import app + with app.app_context(): + try: + meeting = Meeting.query.get(meeting_id) + if not meeting or not meeting.transcript: + self.update_state(state='FAILURE', meta={'error': 'Meeting or transcript not found'}) + return {'status': 'Error', 'error': 'Meeting or transcript not found'} + + api_key = app.config.get("DIFY_SUMMARIZER_API_KEY") + plain_transcript = re.sub(r'^(\s*\[.*?\])\s*', '', meeting.transcript, flags=re.MULTILINE) + + if not plain_transcript.strip(): + self.update_state(state='FAILURE', meta={'error': 'Transcript is empty.'}) + return {'status': 'Error', 'error': 'Transcript is empty'} + + response = ask_dify(api_key, plain_transcript) + summary = response.get("answer") + + if not summary: + self.update_state(state='FAILURE', meta={'error': 'Dify returned empty summary.'}) + return {'status': 'Error', 'error': 'Dify returned empty summary.'} + + meeting.summary = summary + db.session.commit() + + return {'status': 'Success', 'summary': summary} + except Exception as e: + db.session.rollback() + self.update_state( + state='FAILURE', + meta={'exc_type': type(e).__name__, 'exc_message': str(e)} + ) + return {'status': 'Error', 'error': str(e)} + +@celery.task(base=ProgressTask, bind=True) +def preview_action_items_task(self, text_content): + from app import app + with app.app_context(): + try: + self.update_progress(10, 100, "Requesting Dify for action items...") + api_key = app.config.get("DIFY_ACTION_EXTRACTOR_API_KEY") + plain_text = re.sub(r'^(\s*\[.*?\])\s*', '', text_content, flags=re.MULTILINE) + response = ask_dify(api_key, plain_text) + answer_text = response.get("answer", "") + self.update_progress(80, 100, "Parsing response...") + parsed_items = [] + try: + match = re.search(r'\[.*\]', answer_text, re.DOTALL) + if match: + json_str = match.group(0) + parsed_items = json.loads(json_str) + if not isinstance(parsed_items, list): + parsed_items = [] + except (json.JSONDecodeError, TypeError): + parsed_items = [] + self.update_progress(100, 100, "Action item preview generated.") + return {'status': 'Success', 'parsed_items': parsed_items} + except Exception as e: + self.update_state( + state='FAILURE', + meta={'exc_type': type(e).__name__, 'exc_message': str(e)} + ) + +@celery.task(bind=True) +def process_meeting_flow(self, meeting_id, target_language=None): + from app import app + with app.app_context(): + meeting = Meeting.query.get(meeting_id) + if not meeting: + return {'status': 'Error', 'error': 'Meeting not found'} + try: + meeting.status = 'processing' + db.session.commit() + + upload_folder = app.config['UPLOAD_FOLDER'] + # Assuming filename exists on the meeting object + audio_path = os.path.join(upload_folder, meeting.filename.replace('.mp4', '.wav')) + + # The call to transcribe_audio_task needs to be updated as it no longer takes 'language' + transcript_result = transcribe_audio_task.apply(args=[audio_path]).get() + if transcript_result.get('status') != 'Success': + raise Exception(f"Transcription failed: {transcript_result.get('error', 'Unknown error')}") + + meeting.transcript = transcript_result['content'] + db.session.commit() + + if target_language: + translation_result = translate_text_task.apply(args=[meeting.transcript, target_language]).get() + if translation_result.get('status') != 'Success': + raise Exception(f"Translation failed: {translation_result.get('error', 'Unknown error')}") + meeting.translated_transcript = translation_result['content'] + db.session.commit() + + summary_result = summarize_text_task.apply(args=[meeting.id]).get() + if summary_result.get('status') != 'Success': + raise Exception(f"Summarization failed: {summary_result.get('error', 'Unknown error')}") + + meeting.summary = summary_result['summary'] + meeting.status = 'completed' + db.session.commit() + return {'status': 'Success', 'meeting_id': meeting.id} + except Exception as e: + meeting.status = 'failed' + db.session.commit() + return {'status': 'Error', 'error': str(e)} \ No newline at end of file diff --git a/templates/index.html b/templates/index.html new file mode 100644 index 0000000..b37dd5d --- /dev/null +++ b/templates/index.html @@ -0,0 +1,168 @@ + + + + + + AI Meeting Assistant + + + + + +
+
+

AI 會議助手

+

一個強大的工具,用於轉錄、翻譯和總結您的會議內容。

+
+ +
+
+ +
+
+
+ +
+
影片轉音訊 (.wav)
+

從影片檔案中提取音軌,以便進行後續處理。

+
+ + +
+ +
+ + +
+
音訊轉文字 (Whisper)
+

將音訊檔案轉錄成帶有時間戳的逐字稿。

+
+ + +
+
+
+ + +
+
+
+ + +
+ +
+ + +
+
逐段翻譯 (Dify)
+

將逐字稿檔案進行逐段對照翻譯。

+
+ + +
+
+ + +
+ +
+ + +
+
會議結論整理 (Dify)
+

從逐字稿或貼上的文字中生成會議摘要。

+
+ + +
+
+ + +
+
+ + +
+ +
+
+
+
+ + + + + +
+ + + + +