From 9b03456f04883a4d004407ff03e736f53df2a4e9 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 28 Sep 2024 13:28:59 -0400 Subject: [PATCH 01/27] Add style guide and link to notion doc in README --- README.md | 179 +++++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 177 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 823d0b6..841b562 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,177 @@ -# GS-Onboarding -Ground station onboarding for the firmware sub-team +# GS Onboarding Challenge +Welcome to Orbital's Ground Station Onboarding Challenge! Please visit this [Notion doc](https://www.notion.so/uworbital/Ground-Station-Onboarding-10f8a26d767780d7ae8de921d9782b77) for the challenge instructions. Remember to follow our style guide which is written below. + +## Python Style Guide + +- We will be following the Python language style guide [PEP8](https://peps.python.org/pep-0008/) +- If there are any discrepancies between this style guide and PEP8, this style guide takes precedence. + +### Type Hinting Convention + +All function and method parameters (except for the `self` and `cls` parameters) and return signatures should be type hinted. + +```python +def my_add(num1: int, num2: int) -> int: + """ + @brief Adds two numbers together + + @param num1 - The first number to add. + @param num2 - The second number to add. + @return Returns the sum of the two numbers. + """ + return num1 + num2 +``` + +### Comments + +### Naming Conventions + +- `variable_names`, `field_names` and `function_constants` in snake_case +- `_private_field_names`, and `_private_method_names()` in \_snake_case +- `function_names()` and `method_names()` in snake_case +- `CONSTANT_NAMES: Final` and `ENUM_OPTIONS` in CAPITAL_SNAKE_CASE for module and class constants (not for local constant) +- `file_names` in snake_case +- `ClassName` in PascalCase + ```python + # For brevity, the class comments were removed but they should be in real code + import dataclasses + + @dataclasses.dataclass + class PointTwoDimension: + x: int + y: int + + class PointTwoDimension: + def __init__(x: int, y: int): + self.x = x + self.y = y + ``` + +- `EnumName` in PascalCase + + ```python + import enum + + class ErrorCode(enum.Enum): + SUCCESS = 0 + INVALID_ARG = 1 + + # Accessing: + ErrorCode.SUCCESS # + ErrorCode.INVALID_ARG # + ``` + +#### Single Line Comments + +Variable and function names should be descriptive enough to understand even without comments. Comments are needed to describe any complicated logic. Use `#` for single-line comments. + +#### Function and Method Comments + +Function and method comments using `""" """` should exist below the function declaration. For methods, the `self` or `cls` parameter does not require a description. + +```python +def my_add(num1: int, num2: int) -> int: + """ + @brief Adds two numbers together + + @param num1 - The first number to add. + @param num2 - The second number to add. + @return Returns the sum of the two numbers. + """ + return num1 + num2 +``` + +```python +def increase_x(self, count: int) -> None: + """ + @brief Increases the x attribute by the count. + + @param count - Count to increase the x attribute by. + """ + self.x += count +``` + +#### File Header Comments + +File comments are not required + +#### Class Comments + +- Class comments should exist after the class definition +- Provide a brief description given class purpose +- Provide a section in the class comment listing the attributes, their type and purpose +- Enum class comments do not require listing the attributes + +```python +class PointTwoDimension: + """ + @brief Class for storing a 2D point + @attribute x (int) - x coordinate of the point + @attribute y (int) - y coordinate of the point + """ + + def __init__(x: int, y: int): + self.x = x + self.y = y + +@dataclasses.dataclass +class PointTwoDimension: + """ + @brief Class for storing a 2D point + @attribute x (int) - x coordinate of the point + @attribute y (int) - y coordinate of the point + """ + + x: int + y: int +``` + ```python +import enum + +# No comments required +class ErrorCode(enum.Enum): + """ + @brief Enum for the error codes + """ + + SUCCESS = 0 + INVALID_ARG = 1 +``` + +### Imports + +#### Grouping Imports + +Handled by pre-commit + +#### Notes about imports + +- Imports should only be used at the top of the file (no function or scoped imports) +- Only modules should be imported + +```python +# module1 contains very_long_module_name and function foo and variable var. +# very_long_module_name contains bar + +# Yes: +from module1 import very_long_module_name as module2 # Casting to shorter name +import module1 + +module1.foo() +module1.var +module2.bar() + +# No: +from module1.very_long_module_name import bar +from module1 import foo, var + +foo() +var +bar() +``` + +### Other Style Guide Points + +- Only imports, function, class, and constants declarations and the `if __name__ == '__main__'` should be in module scope +- Entry point to a script or program should be through the `main` function +- Add a trailing comma after elements of a list, if you wish to make/preserve each element on a separate line From 6960a2b83936e9599628f017916c6d71c1f5af7f Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 28 Sep 2024 13:38:32 -0400 Subject: [PATCH 02/27] Edit .gitignore --- .gitignore | 45 ++++++++++++++++++++++++++++++++++----------- 1 file changed, 34 insertions(+), 11 deletions(-) diff --git a/.gitignore b/.gitignore index 82f9275..7949c3c 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,5 @@ +### Backend: + # Byte-compiled / optimized / DLL files __pycache__/ *.py[cod] @@ -55,16 +57,6 @@ cover/ *.mo *.pot -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - # Scrapy stuff: .scrapy @@ -159,4 +151,35 @@ cython_debug/ # be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore # and can be added to the global gitignore or merged into this file. For a more nuclear # option (not recommended) you can uncomment the following to ignore the entire idea folder. -#.idea/ +.idea/ + +# IDEs +compile_commands.json +.cache/ + +### Frontend: +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? + +# Dependencies +*node_modules +*.pnp +.pnp.js + +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* +lerna-debug.log* + +# Build +*dist +*dist-ssr +*.local + From a4b9cb5a1a8b54827206b7b679ddd890ff8333a2 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 28 Sep 2024 14:06:08 -0400 Subject: [PATCH 03/27] Setup python --- pyproject.toml | 8 ++++++++ requirements.txt | 8 ++++++++ setup.cfg | 20 ++++++++++++++++++++ setup.py | 4 ++++ 4 files changed, 40 insertions(+) create mode 100644 pyproject.toml create mode 100644 requirements.txt create mode 100644 setup.cfg create mode 100644 setup.py diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..1814a68 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,8 @@ +[build-system] +requires = ["setuptools>=59.0", "wheel"] + +build-backend = "setuptools.build_meta" + +[tool.pytest.ini_options] +addopts = "--cov=gs --cov=obc/tools/python -v" +testpaths = ["test/backend"] diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..75f8284 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,8 @@ +# Regular dependencies +fastapi==0.115.0 +sqlmodel==0.0.22 + +# Development dependencies +httpx==0.27.2 +pytest==7.4.0 +pytest-cov==4.1.0 diff --git a/setup.cfg b/setup.cfg new file mode 100644 index 0000000..bfcfe66 --- /dev/null +++ b/setup.cfg @@ -0,0 +1,20 @@ +[metadata] +name = GS-Onboarding +description = Ground Station Onboarding for the University of Waterloo Orbital Design Team +author = University of Waterloo Orbital Design Team +classifiers = + Programming Language :: Python :: 3.10 :: Only + +[options] +packages = backend +install_requires = + fastapi>=0.115.0 + sqlmodel>=0.0.22 +python_requires = ~=3.10 +zip_safe = no + +[options.extras_require] +testing = + pytest>=7.4 + pytest-cov>=4.1 + diff --git a/setup.py b/setup.py new file mode 100644 index 0000000..7f1a176 --- /dev/null +++ b/setup.py @@ -0,0 +1,4 @@ +from setuptools import setup + +if __name__ == "__main__": + setup() From bc0037e089623a7da7598bc4c902df37c587cc61 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 28 Sep 2024 14:21:31 -0400 Subject: [PATCH 04/27] Create structure of backend --- backend/__init__.py | 0 backend/api/__init__.py | 0 backend/api/endpoint.py | 8 ++++++++ backend/api/request_model.py | 0 backend/api/response_model.py | 0 backend/data/__init__.py | 0 backend/data/base_model.py | 0 backend/data/data_model.py | 0 backend/domain/__init__.py | 0 backend/main.py | 5 +++++ requirements.txt | 2 +- test/backend/__init__.py | 0 test/backend/test_api.py | 0 test/backend/test_data.py | 0 test/backend/test_domain.py | 0 15 files changed, 14 insertions(+), 1 deletion(-) create mode 100644 backend/__init__.py create mode 100644 backend/api/__init__.py create mode 100644 backend/api/endpoint.py create mode 100644 backend/api/request_model.py create mode 100644 backend/api/response_model.py create mode 100644 backend/data/__init__.py create mode 100644 backend/data/base_model.py create mode 100644 backend/data/data_model.py create mode 100644 backend/domain/__init__.py create mode 100644 backend/main.py create mode 100644 test/backend/__init__.py create mode 100644 test/backend/test_api.py create mode 100644 test/backend/test_data.py create mode 100644 test/backend/test_domain.py diff --git a/backend/__init__.py b/backend/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/api/__init__.py b/backend/api/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/api/endpoint.py b/backend/api/endpoint.py new file mode 100644 index 0000000..3915c78 --- /dev/null +++ b/backend/api/endpoint.py @@ -0,0 +1,8 @@ +from fastapi import APIRouter + +resource = APIRouter() + + +@resource.get("/") +async def get(): + return {"Hello": "World"} diff --git a/backend/api/request_model.py b/backend/api/request_model.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/api/response_model.py b/backend/api/response_model.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/data/__init__.py b/backend/data/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/data/base_model.py b/backend/data/base_model.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/data/data_model.py b/backend/data/data_model.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/domain/__init__.py b/backend/domain/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/main.py b/backend/main.py new file mode 100644 index 0000000..c00c6b3 --- /dev/null +++ b/backend/main.py @@ -0,0 +1,5 @@ +from backend.api.endpoint import resource +from fastapi import FastAPI + +app = FastAPI() +app.include_router(router=resource) diff --git a/requirements.txt b/requirements.txt index 75f8284..21c0869 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,5 @@ # Regular dependencies -fastapi==0.115.0 +fastapi[standard]==0.115.0 sqlmodel==0.0.22 # Development dependencies diff --git a/test/backend/__init__.py b/test/backend/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/test/backend/test_api.py b/test/backend/test_api.py new file mode 100644 index 0000000..e69de29 diff --git a/test/backend/test_data.py b/test/backend/test_data.py new file mode 100644 index 0000000..e69de29 diff --git a/test/backend/test_domain.py b/test/backend/test_domain.py new file mode 100644 index 0000000..e69de29 From e1238ce6ab3ba9ec2bb8df3622673a1193ccb6ba Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 29 Sep 2024 13:40:08 -0400 Subject: [PATCH 05/27] Continue --- backend/api/endpoint.py | 8 ----- backend/api/endpoints.py | 34 +++++++++++++++++++ backend/api/middlewares.py | 33 ++++++++++++++++++ backend/api/request_models.py | 7 ++++ backend/api/response_models.py | 7 ++++ .../request_model.py => data/base_models.py} | 0 backend/data/data_model.py | 0 .../response_model.py => data/data_models.py} | 0 backend/data/{base_model.py => enums.py} | 0 backend/main.py | 18 ++++++++-- 10 files changed, 97 insertions(+), 10 deletions(-) delete mode 100644 backend/api/endpoint.py create mode 100644 backend/api/endpoints.py create mode 100644 backend/api/middlewares.py create mode 100644 backend/api/request_models.py create mode 100644 backend/api/response_models.py rename backend/{api/request_model.py => data/base_models.py} (100%) delete mode 100644 backend/data/data_model.py rename backend/{api/response_model.py => data/data_models.py} (100%) rename backend/data/{base_model.py => enums.py} (100%) diff --git a/backend/api/endpoint.py b/backend/api/endpoint.py deleted file mode 100644 index 3915c78..0000000 --- a/backend/api/endpoint.py +++ /dev/null @@ -1,8 +0,0 @@ -from fastapi import APIRouter - -resource = APIRouter() - - -@resource.get("/") -async def get(): - return {"Hello": "World"} diff --git a/backend/api/endpoints.py b/backend/api/endpoints.py new file mode 100644 index 0000000..dcafd63 --- /dev/null +++ b/backend/api/endpoints.py @@ -0,0 +1,34 @@ +from fastapi import APIRouter + +resource = APIRouter() + + +@resource.get("/") +async def get_items(): + """ + Gets all the items + + @return Returns a list of items + """ + return {"Hello": "World"} + + +@resource.post("/") +async def create_item(payload): + """ + Creates an item with the given payload and returns the payload with some other information + + @param payload: The data used to create an item + @return returns the data with some other information + """ + # TODO: Implement this endpoint + + +@resource.delete("/{id}") +async def delete_item(id: int): + """ + Deletes the item with the given id if it exists. Otherwise raises a 404 error. + + @param id: The id of the item to delete + """ + # TODO: Implement this endpoint diff --git a/backend/api/middlewares.py b/backend/api/middlewares.py new file mode 100644 index 0000000..2964fba --- /dev/null +++ b/backend/api/middlewares.py @@ -0,0 +1,33 @@ +from collections.abc import Callable +from typing import Any +from fastapi import FastAPI, Request, Response +from fastapi.middleware.cors import CORSMiddleware +from starlette.middleware.base import BaseHTTPMiddleware + + +def add_cors_middleware(app: FastAPI) -> None: + app.add_middleware( + CORSMiddleware, + allow_origins=["http://localhost:*"], + allow_credentials=True, + allow_methods=["*"], + allow_headers=["*"], + ) + + +# TODO: Implement this logging middleware +class LoggerMiddleware(BaseHTTPMiddleware): + async def dispatch( + self, request: Request, call_next: Callable[[Request], Any] + ) -> Response: + """ + Logs all incoming and outgoing request, response pairs. This method logs the request params, + datetime of request, length of time of request and response. + + @param request: Request received to this middleware + @param call_next: Endpoint or next middleware to be called + @return Response from endpoint + """ + # TODO: Finish implementing this method + response = await call_next(request) + return response diff --git a/backend/api/request_models.py b/backend/api/request_models.py new file mode 100644 index 0000000..56c30b7 --- /dev/null +++ b/backend/api/request_models.py @@ -0,0 +1,7 @@ +from pydantic import BaseModel + + +class RequestModel(BaseModel): + """""" + + # TODO: Need to fill this in diff --git a/backend/api/response_models.py b/backend/api/response_models.py new file mode 100644 index 0000000..84264ee --- /dev/null +++ b/backend/api/response_models.py @@ -0,0 +1,7 @@ +from pydantic import BaseModel + + +class ResponseModel(BaseModel): + """""" + + # TODO: Need to fill this in diff --git a/backend/api/request_model.py b/backend/data/base_models.py similarity index 100% rename from backend/api/request_model.py rename to backend/data/base_models.py diff --git a/backend/data/data_model.py b/backend/data/data_model.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/api/response_model.py b/backend/data/data_models.py similarity index 100% rename from backend/api/response_model.py rename to backend/data/data_models.py diff --git a/backend/data/base_model.py b/backend/data/enums.py similarity index 100% rename from backend/data/base_model.py rename to backend/data/enums.py diff --git a/backend/main.py b/backend/main.py index c00c6b3..0c2630b 100644 --- a/backend/main.py +++ b/backend/main.py @@ -1,5 +1,19 @@ -from backend.api.endpoint import resource +from backend.api.endpoints import resource +from backend.api.middlewares import add_cors_middleware, LoggerMiddleware from fastapi import FastAPI + +def setup_routes(app: FastAPI) -> None: + """Adds the routes to the app""" + app.include_router(router=resource) + + +def setup_middlewares(app: FastAPI) -> None: + """Adds the middlewares to the app""" + add_cors_middleware(app) + app.add_middleware(LoggerMiddleware) + + app = FastAPI() -app.include_router(router=resource) +setup_routes(app) +setup_middlewares(app) From ca3fa7f4118feb50d44d0da4d4dbd23f395698bd Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Wed, 2 Oct 2024 18:46:48 -0400 Subject: [PATCH 06/27] Create models --- backend/api/endpoints.py | 10 +++++++--- backend/api/request_models.py | 9 ++++++--- backend/api/response_models.py | 10 +++++++--- backend/data/base_models.py | 0 backend/data/data_models.py | 36 ++++++++++++++++++++++++++++++++++ backend/data/enums.py | 17 ++++++++++++++++ 6 files changed, 73 insertions(+), 9 deletions(-) delete mode 100644 backend/data/base_models.py diff --git a/backend/api/endpoints.py b/backend/api/endpoints.py index dcafd63..b06accb 100644 --- a/backend/api/endpoints.py +++ b/backend/api/endpoints.py @@ -1,9 +1,13 @@ from fastapi import APIRouter +from backend.api.request_models import CommandRequest +from backend.api.response_models import CommandListResponse +from backend.data.data_models import Command + resource = APIRouter() -@resource.get("/") +@resource.get("/", response_model=CommandListResponse) async def get_items(): """ Gets all the items @@ -13,8 +17,8 @@ async def get_items(): return {"Hello": "World"} -@resource.post("/") -async def create_item(payload): +@resource.post("/", response_model=Command) +async def create_item(payload: CommandRequest): """ Creates an item with the given payload and returns the payload with some other information diff --git a/backend/api/request_models.py b/backend/api/request_models.py index 56c30b7..af22e47 100644 --- a/backend/api/request_models.py +++ b/backend/api/request_models.py @@ -1,7 +1,10 @@ from pydantic import BaseModel -class RequestModel(BaseModel): - """""" +class CommandRequest(BaseModel): + """ + Model representing the command to be created + """ - # TODO: Need to fill this in + command_type: int + params: str | None = None diff --git a/backend/api/response_models.py b/backend/api/response_models.py index 84264ee..6befa14 100644 --- a/backend/api/response_models.py +++ b/backend/api/response_models.py @@ -1,7 +1,11 @@ from pydantic import BaseModel +from backend.data.data_models import Command -class ResponseModel(BaseModel): - """""" - # TODO: Need to fill this in +class CommandListResponse(BaseModel): + """ + List of all commands + """ + + data: list[Command] diff --git a/backend/data/base_models.py b/backend/data/base_models.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/data/data_models.py b/backend/data/data_models.py index e69de29..2cbb360 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -0,0 +1,36 @@ +# Data models used in the onboarding +# NOTE: This file should not be modified +from datetime import datetime +from sqlmodel import Field, SQLModel + +from backend.data.enums import CommandStatus + + +class MainCommand(SQLModel, table=True): + """ + Main command model. + This table represents all the possible commands that can be issued. + + List of commands: https://docs.google.com/spreadsheets/d/1XWXgp3--NHZ4XlxOyBYPS-M_LOU_ai-I6TcvotKhR1s/edit?gid=564815068#gid=564815068 + """ + + id: int | None = Field(default=None, primary_key=True) + name: str + params: str | None = None + format: str | None = None + data_size: int + total_size: int + + +class Command(SQLModel, table=True): + """ + An instance of a MainCommand. + This table holds the data related to actual commands sent from the ground station up to the OBC. + """ + + id: int | None = Field(primary_key=True) + command_type: int = Field(foreign_key=MainCommand.id) + status: CommandStatus = CommandStatus.PENDING + params: str | None = None + created_on: datetime = datetime.now() + updated_on: datetime = datetime.now() diff --git a/backend/data/enums.py b/backend/data/enums.py index e69de29..dba070b 100644 --- a/backend/data/enums.py +++ b/backend/data/enums.py @@ -0,0 +1,17 @@ +# Enums used in the onboarding challenge +from enum import Enum, auto + + +class CommandStatus(Enum): + """ + Enum representing the command status. + + @warning This enum shouldn't be modified + """ + + PENDING = auto() + SCHEDULED = auto() + ONGOING = auto() + CANCELLED = auto() + FAILED = auto() + COMPLETED = auto() From decfd4d5608feea841cd17505affeb61aef8def8 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Wed, 2 Oct 2024 20:44:32 -0400 Subject: [PATCH 07/27] Add model validator for MainCommand --- backend/data/data_models.py | 12 ++++++++++++ test/backend/conftest.py | 39 +++++++++++++++++++++++++++++++++++++ 2 files changed, 51 insertions(+) create mode 100644 test/backend/conftest.py diff --git a/backend/data/data_models.py b/backend/data/data_models.py index 2cbb360..dd289a5 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -1,6 +1,8 @@ # Data models used in the onboarding # NOTE: This file should not be modified from datetime import datetime +from typing import Self +from pydantic import model_validator from sqlmodel import Field, SQLModel from backend.data.enums import CommandStatus @@ -21,6 +23,16 @@ class MainCommand(SQLModel, table=True): data_size: int total_size: int + @model_validator(mode="after") + def validate_params_format(self) -> Self: + """Check that params and format are both None or that the params and format have the same number of comma seperated values""" + if self.params is None and self.format is None: + return self + assert self.params is not None + assert self.format is not None + assert len(self.params.split(",")) == len(self.format.split(",")) + return self + class Command(SQLModel, table=True): """ diff --git a/test/backend/conftest.py b/test/backend/conftest.py new file mode 100644 index 0000000..ea49673 --- /dev/null +++ b/test/backend/conftest.py @@ -0,0 +1,39 @@ +from inspect import formatannotation +from sqlmodel import SQLModel, Session, create_engine +import pytest + +from backend.data.data_models import MainCommand, Command + + +@pytest.fixture +def db_engine(): + sqlite_file_name = "sqlite://" # In memory db for testing + engine = create_engine(sqlite_file_name) + return engine + + +@pytest.fixture(scope="session", autouse=True) +def setup_db(db_engine): + SQLModel.metadata.create_all(db_engine) + with Session(db_engine) as session: + session.add( + MainCommand( + id=1, + name="RTC Sync", + params="time", + format="int 7 bytes", + data_size=7, + total_size=8, + ) + ) + session.add( + MainCommand( + id=2, + name="Manually activate an emergency mode for a specified amount of time", + params="mode_state_number,time", + format="int 1 byte, int 7 bytes", + data_size=8, + total_size=9, + ) + ) + session.commit() From 2357d02cc26766cae2df33eba99f957e98597a58 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Thu, 3 Oct 2024 19:46:01 -0400 Subject: [PATCH 08/27] Init frontend --- frontend/.gitignore | 24 +++++++++++++ frontend/README.md | 50 ++++++++++++++++++++++++++ frontend/eslint.config.js | 28 +++++++++++++++ frontend/index.html | 13 +++++++ frontend/package.json | 29 +++++++++++++++ frontend/public/vite.svg | 1 + frontend/src/App.css | 42 ++++++++++++++++++++++ frontend/src/App.tsx | 35 ++++++++++++++++++ frontend/src/assets/react.svg | 1 + frontend/src/index.css | 68 +++++++++++++++++++++++++++++++++++ frontend/src/main.tsx | 10 ++++++ frontend/src/vite-env.d.ts | 1 + frontend/tsconfig.app.json | 24 +++++++++++++ frontend/tsconfig.json | 7 ++++ frontend/tsconfig.node.json | 22 ++++++++++++ frontend/vite.config.ts | 7 ++++ 16 files changed, 362 insertions(+) create mode 100644 frontend/.gitignore create mode 100644 frontend/README.md create mode 100644 frontend/eslint.config.js create mode 100644 frontend/index.html create mode 100644 frontend/package.json create mode 100644 frontend/public/vite.svg create mode 100644 frontend/src/App.css create mode 100644 frontend/src/App.tsx create mode 100644 frontend/src/assets/react.svg create mode 100644 frontend/src/index.css create mode 100644 frontend/src/main.tsx create mode 100644 frontend/src/vite-env.d.ts create mode 100644 frontend/tsconfig.app.json create mode 100644 frontend/tsconfig.json create mode 100644 frontend/tsconfig.node.json create mode 100644 frontend/vite.config.ts 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..74872fd --- /dev/null +++ b/frontend/README.md @@ -0,0 +1,50 @@ +# React + TypeScript + Vite + +This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules. + +Currently, two official plugins are available: + +- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react/README.md) uses [Babel](https://babeljs.io/) for Fast Refresh +- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh + +## Expanding the ESLint configuration + +If you are developing a production application, we recommend updating the configuration to enable type aware lint rules: + +- Configure the top-level `parserOptions` property like this: + +```js +export default tseslint.config({ + languageOptions: { + // other options... + parserOptions: { + project: ['./tsconfig.node.json', './tsconfig.app.json'], + tsconfigRootDir: import.meta.dirname, + }, + }, +}) +``` + +- Replace `tseslint.configs.recommended` to `tseslint.configs.recommendedTypeChecked` or `tseslint.configs.strictTypeChecked` +- Optionally add `...tseslint.configs.stylisticTypeChecked` +- Install [eslint-plugin-react](https://github.com/jsx-eslint/eslint-plugin-react) and update the config: + +```js +// eslint.config.js +import react from 'eslint-plugin-react' + +export default tseslint.config({ + // Set the react version + settings: { react: { version: '18.3' } }, + plugins: { + // Add the react plugin + react, + }, + rules: { + // other rules... + // Enable its recommended rules + ...react.configs.recommended.rules, + ...react.configs['jsx-runtime'].rules, + }, +}) +``` diff --git a/frontend/eslint.config.js b/frontend/eslint.config.js new file mode 100644 index 0000000..092408a --- /dev/null +++ b/frontend/eslint.config.js @@ -0,0 +1,28 @@ +import js from '@eslint/js' +import globals from 'globals' +import reactHooks from 'eslint-plugin-react-hooks' +import reactRefresh from 'eslint-plugin-react-refresh' +import tseslint from 'typescript-eslint' + +export default tseslint.config( + { ignores: ['dist'] }, + { + extends: [js.configs.recommended, ...tseslint.configs.recommended], + files: ['**/*.{ts,tsx}'], + languageOptions: { + ecmaVersion: 2020, + globals: globals.browser, + }, + plugins: { + 'react-hooks': reactHooks, + 'react-refresh': reactRefresh, + }, + rules: { + ...reactHooks.configs.recommended.rules, + 'react-refresh/only-export-components': [ + 'warn', + { allowConstantExport: true }, + ], + }, + }, +) diff --git a/frontend/index.html b/frontend/index.html new file mode 100644 index 0000000..e4b78ea --- /dev/null +++ b/frontend/index.html @@ -0,0 +1,13 @@ + + + + + + + Vite + React + TS + + +
+ + + diff --git a/frontend/package.json b/frontend/package.json new file mode 100644 index 0000000..f5a179a --- /dev/null +++ b/frontend/package.json @@ -0,0 +1,29 @@ +{ + "name": "frontend", + "private": true, + "version": "0.0.0", + "type": "module", + "scripts": { + "dev": "vite", + "build": "tsc -b && vite build", + "lint": "eslint .", + "preview": "vite preview" + }, + "dependencies": { + "react": "^18.3.1", + "react-dom": "^18.3.1" + }, + "devDependencies": { + "@eslint/js": "^9.9.0", + "@types/react": "^18.3.3", + "@types/react-dom": "^18.3.0", + "@vitejs/plugin-react-swc": "^3.5.0", + "eslint": "^9.9.0", + "eslint-plugin-react-hooks": "^5.1.0-rc.0", + "eslint-plugin-react-refresh": "^0.4.9", + "globals": "^15.9.0", + "typescript": "^5.5.3", + "typescript-eslint": "^8.0.1", + "vite": "^5.4.1" + } +} 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.tsx b/frontend/src/App.tsx new file mode 100644 index 0000000..afe48ac --- /dev/null +++ b/frontend/src/App.tsx @@ -0,0 +1,35 @@ +import { useState } from 'react' +import reactLogo from './assets/react.svg' +import viteLogo from '/vite.svg' +import './App.css' + +function App() { + const [count, setCount] = useState(0) + + return ( + <> +
+ + Vite logo + + + React logo + +
+

Vite + React

+
+ +

+ Edit src/App.tsx and save to test HMR +

+
+

+ Click on the Vite and React logos to learn more +

+ + ) +} + +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/index.css b/frontend/src/index.css new file mode 100644 index 0000000..6119ad9 --- /dev/null +++ b/frontend/src/index.css @@ -0,0 +1,68 @@ +:root { + font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif; + line-height: 1.5; + font-weight: 400; + + color-scheme: light dark; + color: rgba(255, 255, 255, 0.87); + background-color: #242424; + + font-synthesis: none; + text-rendering: optimizeLegibility; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +a { + font-weight: 500; + color: #646cff; + text-decoration: inherit; +} +a:hover { + color: #535bf2; +} + +body { + margin: 0; + display: flex; + place-items: center; + min-width: 320px; + min-height: 100vh; +} + +h1 { + font-size: 3.2em; + line-height: 1.1; +} + +button { + border-radius: 8px; + border: 1px solid transparent; + padding: 0.6em 1.2em; + font-size: 1em; + font-weight: 500; + font-family: inherit; + background-color: #1a1a1a; + cursor: pointer; + transition: border-color 0.25s; +} +button:hover { + border-color: #646cff; +} +button:focus, +button:focus-visible { + outline: 4px auto -webkit-focus-ring-color; +} + +@media (prefers-color-scheme: light) { + :root { + color: #213547; + background-color: #ffffff; + } + a:hover { + color: #747bff; + } + button { + background-color: #f9f9f9; + } +} diff --git a/frontend/src/main.tsx b/frontend/src/main.tsx new file mode 100644 index 0000000..6f4ac9b --- /dev/null +++ b/frontend/src/main.tsx @@ -0,0 +1,10 @@ +import { StrictMode } from 'react' +import { createRoot } from 'react-dom/client' +import App from './App.tsx' +import './index.css' + +createRoot(document.getElementById('root')!).render( + + + , +) diff --git a/frontend/src/vite-env.d.ts b/frontend/src/vite-env.d.ts new file mode 100644 index 0000000..11f02fe --- /dev/null +++ b/frontend/src/vite-env.d.ts @@ -0,0 +1 @@ +/// diff --git a/frontend/tsconfig.app.json b/frontend/tsconfig.app.json new file mode 100644 index 0000000..f0a2350 --- /dev/null +++ b/frontend/tsconfig.app.json @@ -0,0 +1,24 @@ +{ + "compilerOptions": { + "target": "ES2020", + "useDefineForClassFields": true, + "lib": ["ES2020", "DOM", "DOM.Iterable"], + "module": "ESNext", + "skipLibCheck": true, + + /* Bundler mode */ + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "isolatedModules": true, + "moduleDetection": "force", + "noEmit": true, + "jsx": "react-jsx", + + /* Linting */ + "strict": true, + "noUnusedLocals": true, + "noUnusedParameters": true, + "noFallthroughCasesInSwitch": true + }, + "include": ["src"] +} diff --git a/frontend/tsconfig.json b/frontend/tsconfig.json new file mode 100644 index 0000000..1ffef60 --- /dev/null +++ b/frontend/tsconfig.json @@ -0,0 +1,7 @@ +{ + "files": [], + "references": [ + { "path": "./tsconfig.app.json" }, + { "path": "./tsconfig.node.json" } + ] +} diff --git a/frontend/tsconfig.node.json b/frontend/tsconfig.node.json new file mode 100644 index 0000000..0d3d714 --- /dev/null +++ b/frontend/tsconfig.node.json @@ -0,0 +1,22 @@ +{ + "compilerOptions": { + "target": "ES2022", + "lib": ["ES2023"], + "module": "ESNext", + "skipLibCheck": true, + + /* Bundler mode */ + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "isolatedModules": true, + "moduleDetection": "force", + "noEmit": true, + + /* Linting */ + "strict": true, + "noUnusedLocals": true, + "noUnusedParameters": true, + "noFallthroughCasesInSwitch": true + }, + "include": ["vite.config.ts"] +} diff --git a/frontend/vite.config.ts b/frontend/vite.config.ts new file mode 100644 index 0000000..861b04b --- /dev/null +++ b/frontend/vite.config.ts @@ -0,0 +1,7 @@ +import { defineConfig } from 'vite' +import react from '@vitejs/plugin-react-swc' + +// https://vitejs.dev/config/ +export default defineConfig({ + plugins: [react()], +}) From c0680858dafb5b7e555dab999c17bd391e35fd7e Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Fri, 4 Oct 2024 18:11:12 -0400 Subject: [PATCH 09/27] Restructure api directory --- backend/api/endpoints.py | 4 ++-- backend/api/middlewares/cors_middleware.py | 17 +++++++++++++++++ .../logger_middleware.py} | 13 +------------ .../request_model.py} | 0 .../response_model.py} | 0 backend/data/data_models.py | 11 ++++++++--- backend/main.py | 3 ++- 7 files changed, 30 insertions(+), 18 deletions(-) create mode 100644 backend/api/middlewares/cors_middleware.py rename backend/api/{middlewares.py => middlewares/logger_middleware.py} (70%) rename backend/api/{request_models.py => models/request_model.py} (100%) rename backend/api/{response_models.py => models/response_model.py} (100%) diff --git a/backend/api/endpoints.py b/backend/api/endpoints.py index b06accb..360688d 100644 --- a/backend/api/endpoints.py +++ b/backend/api/endpoints.py @@ -1,7 +1,7 @@ from fastapi import APIRouter -from backend.api.request_models import CommandRequest -from backend.api.response_models import CommandListResponse +from backend.api.models.request_model import CommandRequest +from backend.api.models.response_model import CommandListResponse from backend.data.data_models import Command resource = APIRouter() diff --git a/backend/api/middlewares/cors_middleware.py b/backend/api/middlewares/cors_middleware.py new file mode 100644 index 0000000..11f14f7 --- /dev/null +++ b/backend/api/middlewares/cors_middleware.py @@ -0,0 +1,17 @@ +from fastapi import FastAPI +from fastapi.middleware.cors import CORSMiddleware + + +def add_cors_middleware(app: FastAPI) -> None: + """ + Adds the cors middleware to the FastAPI app + + @param app: FastAPI app to add the middleware to + """ + app.add_middleware( + CORSMiddleware, + allow_origins=["http://localhost:*"], + allow_credentials=True, + allow_methods=["*"], + allow_headers=["*"], + ) diff --git a/backend/api/middlewares.py b/backend/api/middlewares/logger_middleware.py similarity index 70% rename from backend/api/middlewares.py rename to backend/api/middlewares/logger_middleware.py index 2964fba..c090a35 100644 --- a/backend/api/middlewares.py +++ b/backend/api/middlewares/logger_middleware.py @@ -1,20 +1,9 @@ from collections.abc import Callable from typing import Any -from fastapi import FastAPI, Request, Response -from fastapi.middleware.cors import CORSMiddleware +from fastapi import Request, Response from starlette.middleware.base import BaseHTTPMiddleware -def add_cors_middleware(app: FastAPI) -> None: - app.add_middleware( - CORSMiddleware, - allow_origins=["http://localhost:*"], - allow_credentials=True, - allow_methods=["*"], - allow_headers=["*"], - ) - - # TODO: Implement this logging middleware class LoggerMiddleware(BaseHTTPMiddleware): async def dispatch( diff --git a/backend/api/request_models.py b/backend/api/models/request_model.py similarity index 100% rename from backend/api/request_models.py rename to backend/api/models/request_model.py diff --git a/backend/api/response_models.py b/backend/api/models/response_model.py similarity index 100% rename from backend/api/response_models.py rename to backend/api/models/response_model.py diff --git a/backend/data/data_models.py b/backend/data/data_models.py index dd289a5..4927eb5 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -28,9 +28,14 @@ def validate_params_format(self) -> Self: """Check that params and format are both None or that the params and format have the same number of comma seperated values""" if self.params is None and self.format is None: return self - assert self.params is not None - assert self.format is not None - assert len(self.params.split(",")) == len(self.format.split(",")) + if self.params is None: + raise ValueError("params is None but format is not None") + if self.format is None: + raise ValueError("format is None but params is not None") + if self.params.count(",") != self.format.count(","): + raise ValueError( + "params and format must have the same number of comma seperated values" + ) return self diff --git a/backend/main.py b/backend/main.py index 0c2630b..1034a9e 100644 --- a/backend/main.py +++ b/backend/main.py @@ -1,5 +1,6 @@ from backend.api.endpoints import resource -from backend.api.middlewares import add_cors_middleware, LoggerMiddleware +from backend.api.middlewares.cors_middleware import add_cors_middleware +from backend.api.middlewares.logger_middleware import LoggerMiddleware from fastapi import FastAPI From 4b67126c88ad2e947ceed6e3b5b6cfd54d3121c4 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Fri, 4 Oct 2024 19:17:51 -0400 Subject: [PATCH 10/27] Create test for setup --- backend/data/data_models.py | 7 ++-- pyproject.toml | 2 +- test/backend/conftest.py | 68 ++++++++++++++++++++++++------------- test/backend/test_setup.py | 42 +++++++++++++++++++++++ 4 files changed, 92 insertions(+), 27 deletions(-) create mode 100644 test/backend/test_setup.py diff --git a/backend/data/data_models.py b/backend/data/data_models.py index 4927eb5..0624062 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -1,7 +1,6 @@ # Data models used in the onboarding # NOTE: This file should not be modified from datetime import datetime -from typing import Self from pydantic import model_validator from sqlmodel import Field, SQLModel @@ -24,7 +23,7 @@ class MainCommand(SQLModel, table=True): total_size: int @model_validator(mode="after") - def validate_params_format(self) -> Self: + def validate_params_format(self): """Check that params and format are both None or that the params and format have the same number of comma seperated values""" if self.params is None and self.format is None: return self @@ -46,7 +45,9 @@ class Command(SQLModel, table=True): """ id: int | None = Field(primary_key=True) - command_type: int = Field(foreign_key=MainCommand.id) + command_type: int = Field( + foreign_key="maincommand.id" + ) # Forign key must be a string status: CommandStatus = CommandStatus.PENDING params: str | None = None created_on: datetime = datetime.now() diff --git a/pyproject.toml b/pyproject.toml index 1814a68..07a622e 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -4,5 +4,5 @@ requires = ["setuptools>=59.0", "wheel"] build-backend = "setuptools.build_meta" [tool.pytest.ini_options] -addopts = "--cov=gs --cov=obc/tools/python -v" +addopts = "--cov=backend -v" testpaths = ["test/backend"] diff --git a/test/backend/conftest.py b/test/backend/conftest.py index ea49673..14cc8dd 100644 --- a/test/backend/conftest.py +++ b/test/backend/conftest.py @@ -1,6 +1,7 @@ -from inspect import formatannotation +from _pytest import scope from sqlmodel import SQLModel, Session, create_engine import pytest +from datetime import datetime from backend.data.data_models import MainCommand, Command @@ -12,28 +13,49 @@ def db_engine(): return engine -@pytest.fixture(scope="session", autouse=True) -def setup_db(db_engine): +@pytest.fixture +def unix_time(): + default_datetime = datetime.strptime("2021-01-01 00:00:00", "%Y-%m-%d %H:%M:%S") + return int(default_datetime.timestamp()) + + +@pytest.fixture +def main_commands(): + return [ + MainCommand( + id=1, + name="RTC Sync", + params="time", + format="int 7 bytes", + data_size=7, + total_size=8, + ), + MainCommand( + id=2, + name="Manually activate an emergency mode for a specified amount of time", + params="mode_state_number,time", + format="int 1 byte, int 7 bytes", + data_size=8, + total_size=9, + ), + ] + + +@pytest.fixture +def commands(unix_time): + return [ + Command(id=1, command_type=1, params=f"{unix_time}"), # RTC Sync for 2021-01-01 + Command( + id=2, command_type=2, params=f"1,{unix_time}" + ), # Emergency mode for 2021-01-01 + ] + + +@pytest.fixture(scope="function", autouse=True) +def setup_db(db_engine, main_commands, commands): SQLModel.metadata.create_all(db_engine) with Session(db_engine) as session: - session.add( - MainCommand( - id=1, - name="RTC Sync", - params="time", - format="int 7 bytes", - data_size=7, - total_size=8, - ) - ) - session.add( - MainCommand( - id=2, - name="Manually activate an emergency mode for a specified amount of time", - params="mode_state_number,time", - format="int 1 byte, int 7 bytes", - data_size=8, - total_size=9, - ) - ) + session.add_all(main_commands) + session.commit() + session.add_all(commands) session.commit() diff --git a/test/backend/test_setup.py b/test/backend/test_setup.py new file mode 100644 index 0000000..58b043a --- /dev/null +++ b/test/backend/test_setup.py @@ -0,0 +1,42 @@ +# Test to make sure the conftest.py setup is correct +import pytest +from backend.data.data_models import MainCommand, Command +from sqlmodel import Session + + +@pytest.mark.parametrize( + "command_id, name, params, format, data_size, total_size", + [ + (1, "RTC Sync", "time", "int 7 bytes", 7, 8), + ( + 2, + "Manually activate an emergency mode for a specified amount of time", + "mode_state_number,time", + "int 1 byte, int 7 bytes", + 8, + 9, + ), + ], +) +def test_main_command_setup( + db_engine, command_id, name, params, format, data_size, total_size +): + with Session(db_engine) as session: + main_command = session.get(MainCommand, command_id) + assert main_command + assert main_command.name == name + assert main_command.params == params + assert main_command.format == format + assert main_command.data_size == data_size + assert main_command.total_size == total_size + + +@pytest.mark.parametrize( + "id, command_type, params", [(1, 1, "{unix_time}"), [2, 2, "1,{unix_time}"]] +) +def test_command_setup(db_engine, id, command_type, params, unix_time): + with Session(db_engine) as session: + command = session.get(Command, id) + assert command + assert command.command_type == command_type + assert command.params == params.format(unix_time=unix_time) From 86bb6c94da5106d2ec6d479e91bc16b138718637 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Fri, 4 Oct 2024 21:50:12 -0400 Subject: [PATCH 11/27] Continue on backend --- backend/api/endpoints.py | 11 ++++++++--- backend/api/lifespan.py | 13 +++++++++++++ backend/api/middlewares/__init__.py | 0 backend/api/models/__init__.py | 0 backend/data/engine.py | 10 ++++++++++ backend/domain/base_service.py | 4 ++++ backend/domain/command_service.py | 0 backend/domain/iservice.py | 9 +++++++++ backend/main.py | 3 ++- db.sql | Bin 0 -> 12288 bytes 10 files changed, 46 insertions(+), 4 deletions(-) create mode 100644 backend/api/lifespan.py create mode 100644 backend/api/middlewares/__init__.py create mode 100644 backend/api/models/__init__.py create mode 100644 backend/data/engine.py create mode 100644 backend/domain/base_service.py create mode 100644 backend/domain/command_service.py create mode 100644 backend/domain/iservice.py create mode 100644 db.sql diff --git a/backend/api/endpoints.py b/backend/api/endpoints.py index 360688d..7de7c01 100644 --- a/backend/api/endpoints.py +++ b/backend/api/endpoints.py @@ -1,20 +1,25 @@ -from fastapi import APIRouter +from fastapi import APIRouter, Depends +from sqlmodel import Session, select from backend.api.models.request_model import CommandRequest from backend.api.models.response_model import CommandListResponse from backend.data.data_models import Command +from backend.data.engine import get_db resource = APIRouter() @resource.get("/", response_model=CommandListResponse) -async def get_items(): +async def get_items(db: Session = Depends(get_db)): """ Gets all the items @return Returns a list of items """ - return {"Hello": "World"} + query = select(Command) + items = db.exec(query).all() + print(f"Items: {items}") + return {"data": items} @resource.post("/", response_model=Command) diff --git a/backend/api/lifespan.py b/backend/api/lifespan.py new file mode 100644 index 0000000..addab1f --- /dev/null +++ b/backend/api/lifespan.py @@ -0,0 +1,13 @@ +from contextlib import asynccontextmanager +from fastapi import FastAPI +from sqlmodel import SQLModel +import backend.data.data_models +from backend.data.engine import get_db + + +@asynccontextmanager +async def lifespan(_: FastAPI): + print("Starting the app") + SQLModel.metadata.create_all(get_db().connection()) + yield + print("Stopping the app") diff --git a/backend/api/middlewares/__init__.py b/backend/api/middlewares/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/api/models/__init__.py b/backend/api/models/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/data/engine.py b/backend/data/engine.py new file mode 100644 index 0000000..519fb49 --- /dev/null +++ b/backend/data/engine.py @@ -0,0 +1,10 @@ +from typing import Final +from sqlmodel import Session, create_engine + +SQL_PATH: Final[str] = "sqlite:///db.sql" + + +def get_db() -> Session: + engine = create_engine(SQL_PATH) + with Session(engine) as session: + return session diff --git a/backend/domain/base_service.py b/backend/domain/base_service.py new file mode 100644 index 0000000..cd07543 --- /dev/null +++ b/backend/domain/base_service.py @@ -0,0 +1,4 @@ +from backend.domain.iservice import IService + + +class BaseService(IService): ... diff --git a/backend/domain/command_service.py b/backend/domain/command_service.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/domain/iservice.py b/backend/domain/iservice.py new file mode 100644 index 0000000..a9d0026 --- /dev/null +++ b/backend/domain/iservice.py @@ -0,0 +1,9 @@ +from abc import ABC, abstractmethod +from typing import Generic, TypeVar + +T = TypeVar("T") + + +class IService(ABC, Generic[T]): + @abstractmethod + def get(self, id: int) -> T: ... diff --git a/backend/main.py b/backend/main.py index 1034a9e..82e04b9 100644 --- a/backend/main.py +++ b/backend/main.py @@ -1,4 +1,5 @@ from backend.api.endpoints import resource +from backend.api.lifespan import lifespan from backend.api.middlewares.cors_middleware import add_cors_middleware from backend.api.middlewares.logger_middleware import LoggerMiddleware from fastapi import FastAPI @@ -15,6 +16,6 @@ def setup_middlewares(app: FastAPI) -> None: app.add_middleware(LoggerMiddleware) -app = FastAPI() +app = FastAPI(lifespan=lifespan) setup_routes(app) setup_middlewares(app) diff --git a/db.sql b/db.sql new file mode 100644 index 0000000000000000000000000000000000000000..585abf290650ba6b9d05ea41cfdd8d9ac6d4be62 GIT binary patch literal 12288 zcmeI#O;5rw7zglfm=G{TZydNh35nSfxx*iT?b z1{n$*J&=Est}jp9^!e@7`#U#MA-yD%SSmVYhs-e8IT2&b)U}}NqNL-!#Xz@4uD_{k zvXiIlijJCy!83y2n&5Wh?sXbsVl;FjZ2`ru68D#+4(i zGkazGSWaZTjLr|7gi?h;KN-=bzKGxM@Rc32aj-RhE7~1L^yr57$ch3xJ+s*ryxsEB zIV*o*n*?ui!Mz5*r&va#Wxjf2*?u1w#cI87eD2Soatm@@a|J9H*n&2aakvm!N$xL4 z_Abx((qwe{vOkU9)^n>w$>HYkZ=JH8zRmTc9|ZysfB*y_009U<00Izz00bZafgKjW e`+tXjE>eR41Rwwb2tWV=5P$##AOHafWCB0#5TOA8 literal 0 HcmV?d00001 From bb57c454a9cf4497095e69f94beeee31e4814486 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 5 Oct 2024 11:52:14 -0400 Subject: [PATCH 12/27] Add workflows --- .github/workflows/frontend_build.yml | 26 +++++++++++++++++++++ .github/workflows/pytest.yml | 34 ++++++++++++++++++++++++++++ 2 files changed, 60 insertions(+) create mode 100644 .github/workflows/frontend_build.yml create mode 100644 .github/workflows/pytest.yml diff --git a/.github/workflows/frontend_build.yml b/.github/workflows/frontend_build.yml new file mode 100644 index 0000000..4758d7e --- /dev/null +++ b/.github/workflows/frontend_build.yml @@ -0,0 +1,26 @@ +name: Frontend Build +on: + pull_request: + push: + branches: + - main + +jobs: + build: + + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + - name: Use Node.js + uses: actions/setup-node@v4 + with: + node-version: '20.x' + - name: Install dependencies for frontend + run: | + cd frontend + npm ci + - name: Build project + run: | + cd frontend + npm run build diff --git a/.github/workflows/pytest.yml b/.github/workflows/pytest.yml new file mode 100644 index 0000000..7ecf70d --- /dev/null +++ b/.github/workflows/pytest.yml @@ -0,0 +1,34 @@ +name: Pytest + +on: + pull_request: + push: + branches: + - main + +jobs: + pytest: + runs-on: ${{ matrix.os }} + + strategy: + matrix: + os: [ubuntu-latest, windows-latest] + python-version: ['3.10'] + + steps: + - uses: actions/checkout@v3 + + - name: Set up Python ${{ matrix.python-version }} + uses: actions/setup-python@v3 + with: + python-version: ${{ matrix.python-version }} + + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install -r requirements.txt + pip install -e . + + - name: Run pytest + run: | + python -m pytest From 01339c137a15da86339d638d2552b0823a183204 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 5 Oct 2024 11:56:14 -0400 Subject: [PATCH 13/27] Add PR template --- .github/pull_request_template.md | 12 ++++++++++++ 1 file changed, 12 insertions(+) create mode 100644 .github/pull_request_template.md diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md new file mode 100644 index 0000000..1e93af2 --- /dev/null +++ b/.github/pull_request_template.md @@ -0,0 +1,12 @@ +# Purpose +Completed the GS on-boarding task. Include a screenshot of the front-end of the application. + +# New Changes +- Explain new changes + +# Testing +- Explain tests that you ran to verify code functionality. +- Any functions that can be unit-tested should include a unit test in the PR. Otherwise, explain why it cannot be unit-tested. + +# Outstanding Changes +- If there are non-critical changes (i.e. additional features) that can be made to this feature in the future, indicate them here. From d2069f260766f0d0a27896ffee179285147fc568 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 5 Oct 2024 12:38:50 -0400 Subject: [PATCH 14/27] Add naming convention for frontend and package-lock.json --- README.md | 36 + frontend/package-lock.json | 2595 +++++++++++++++++++++++++++++ frontend/src/{App.css => app.css} | 0 frontend/src/{App.tsx => app.tsx} | 2 +- frontend/src/main.tsx | 2 +- 5 files changed, 2633 insertions(+), 2 deletions(-) create mode 100644 frontend/package-lock.json rename frontend/src/{App.css => app.css} (100%) rename frontend/src/{App.tsx => app.tsx} (97%) diff --git a/README.md b/README.md index 841b562..27d4024 100644 --- a/README.md +++ b/README.md @@ -175,3 +175,39 @@ bar() - Only imports, function, class, and constants declarations and the `if __name__ == '__main__'` should be in module scope - Entry point to a script or program should be through the `main` function - Add a trailing comma after elements of a list, if you wish to make/preserve each element on a separate line + +## Typescript/React Style Guide + +### Comments + +#### Single Line Comments + +Variable and function names should be descriptive enough to understand even without comments. Comments are needed to describe any complicated logic. You may use `//` or `/* */` for single line comments. + +#### Function Comments + +Function comments should follow the format shown below: +```typescript +/** + * @brief Adds two numbers together + * + * @param num1 - The first number to add. + * @param num2 - The second number to add. + * @return Returns the sum of the two numbers. + */ +function addNumbers(num1: number, num2: number): number { + return num1 + num2; +} +``` + +#### File Header Comments + +- File comments are not required + +### ****Naming and typing conventions**** + +- `variableNames` in camelCase +- `functionNames()` in camelCase +- `CONSTANT_NAME` in CAPITAL_SNAKE_CASE +- `file_names` in snake_case +- `ClassName` and `ComponentName` in PascalCase diff --git a/frontend/package-lock.json b/frontend/package-lock.json new file mode 100644 index 0000000..4877c0a --- /dev/null +++ b/frontend/package-lock.json @@ -0,0 +1,2595 @@ +{ + "name": "frontend", + "version": "0.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "frontend", + "version": "0.0.0", + "dependencies": { + "react": "^18.3.1", + "react-dom": "^18.3.1" + }, + "devDependencies": { + "@eslint/js": "^9.9.0", + "@types/react": "^18.3.3", + "@types/react-dom": "^18.3.0", + "@vitejs/plugin-react-swc": "^3.5.0", + "eslint": "^9.9.0", + "eslint-plugin-react-hooks": "^5.1.0-rc.0", + "eslint-plugin-react-refresh": "^0.4.9", + "globals": "^15.9.0", + "typescript": "^5.5.3", + "typescript-eslint": "^8.0.1", + "vite": "^5.4.1" + } + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", + "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", + "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", + "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", + "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", + "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", + "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", + "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", + "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", + "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", + "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", + "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", + "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", + "cpu": [ + "loong64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", + "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", + "cpu": [ + "mips64el" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", + "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", + "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", + "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", + "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", + "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", + "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", + "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", + "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", + "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", + "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "dependencies": { + "eslint-visitor-keys": "^3.3.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "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, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.11.1", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.11.1.tgz", + "integrity": "sha512-m4DVN9ZqskZoLU5GlWZadwDnYo3vAEydiUayB9widCl9ffWx2IvPnp6n3on5rJmziJSw9Bv+Z3ChDVdMwXCY8Q==", + "dev": true, + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/config-array": { + "version": "0.18.0", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.18.0.tgz", + "integrity": "sha512-fTxvnS1sRMu3+JjXwJG0j/i4RT9u4qJ+lqS/yCGap4lH4zZGzQ7tu+xZqQmcMZq5OBZDL4QRxQzRjkWcGt8IVw==", + "dev": true, + "dependencies": { + "@eslint/object-schema": "^2.1.4", + "debug": "^4.3.1", + "minimatch": "^3.1.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/core": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.6.0.tgz", + "integrity": "sha512-8I2Q8ykA4J0x0o7cg67FPVnehcqWTBehu/lmY+bolPFHGjh49YzGBMXTvpqVgEbBdvNCSxj6iFgiIyHzf03lzg==", + "dev": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.1.0.tgz", + "integrity": "sha512-4Bfj15dVJdoy3RfZmmo86RK1Fwzn6SstsvK9JS+BaVKqC6QQQQyXekNaC+g+LKNgkQ+2VhGAzm6hO40AhMR3zQ==", + "dev": true, + "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, + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@eslint/js": { + "version": "9.12.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.12.0.tgz", + "integrity": "sha512-eohesHH8WFRUprDNyEREgqP6beG6htMeUYeCpkEgBCieCMme5r9zFWjzAJp//9S+Kub4rqE+jXe9Cp1a7IYIIA==", + "dev": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/object-schema": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.4.tgz", + "integrity": "sha512-BsWiH1yFGjXXS2yvrf5LyuoSIIbPrGUWob917o+BTKuZ7qJdxX8aJLRxs1fS9n6r7vESrq1OUqb68dANcFXuQQ==", + "dev": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/plugin-kit": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.0.tgz", + "integrity": "sha512-vH9PiIMMwvhCx31Af3HiGzsVNULDbyVkHXwlemn/B0TFj/00ho3y55efXrUZTfQipxoHC5u4xq6zblww1zm1Ig==", + "dev": true, + "dependencies": { + "levn": "^0.4.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@humanfs/core": { + "version": "0.19.0", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.0.tgz", + "integrity": "sha512-2cbWIHbZVEweE853g8jymffCA+NCMiuqeECeBBLm8dg2oFdjuGJhgN4UAbI+6v0CKbbhvtXA4qV8YR5Ji86nmw==", + "dev": true, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node": { + "version": "0.16.5", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.5.tgz", + "integrity": "sha512-KSPA4umqSG4LHYRodq31VDwKAvaTF4xmVlzM8Aeh4PlU1JQ3IG0wiA8C25d3RQ9nJyM3mBHyI53K06VVL/oFFg==", + "dev": true, + "dependencies": { + "@humanfs/core": "^0.19.0", + "@humanwhocodes/retry": "^0.3.0" + }, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "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, + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.24.0.tgz", + "integrity": "sha512-Q6HJd7Y6xdB48x8ZNVDOqsbh2uByBhgK8PiQgPhwkIw/HC/YX5Ghq2mQY5sRMZWHb3VsFkWooUVOZHKr7DmDIA==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.24.0.tgz", + "integrity": "sha512-ijLnS1qFId8xhKjT81uBHuuJp2lU4x2yxa4ctFPtG+MqEE6+C5f/+X/bStmxapgmwLwiL3ih122xv8kVARNAZA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.24.0.tgz", + "integrity": "sha512-bIv+X9xeSs1XCk6DVvkO+S/z8/2AMt/2lMqdQbMrmVpgFvXlmde9mLcbQpztXm1tajC3raFDqegsH18HQPMYtA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.24.0.tgz", + "integrity": "sha512-X6/nOwoFN7RT2svEQWUsW/5C/fYMBe4fnLK9DQk4SX4mgVBiTA9h64kjUYPvGQ0F/9xwJ5U5UfTbl6BEjaQdBQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.24.0.tgz", + "integrity": "sha512-0KXvIJQMOImLCVCz9uvvdPgfyWo93aHHp8ui3FrtOP57svqrF/roSSR5pjqL2hcMp0ljeGlU4q9o/rQaAQ3AYA==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.24.0.tgz", + "integrity": "sha512-it2BW6kKFVh8xk/BnHfakEeoLPv8STIISekpoF+nBgWM4d55CZKc7T4Dx1pEbTnYm/xEKMgy1MNtYuoA8RFIWw==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.24.0.tgz", + "integrity": "sha512-i0xTLXjqap2eRfulFVlSnM5dEbTVque/3Pi4g2y7cxrs7+a9De42z4XxKLYJ7+OhE3IgxvfQM7vQc43bwTgPwA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.24.0.tgz", + "integrity": "sha512-9E6MKUJhDuDh604Qco5yP/3qn3y7SLXYuiC0Rpr89aMScS2UAmK1wHP2b7KAa1nSjWJc/f/Lc0Wl1L47qjiyQw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.24.0.tgz", + "integrity": "sha512-2XFFPJ2XMEiF5Zi2EBf4h73oR1V/lycirxZxHZNc93SqDN/IWhYYSYj8I9381ikUFXZrz2v7r2tOVk2NBwxrWw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.24.0.tgz", + "integrity": "sha512-M3Dg4hlwuntUCdzU7KjYqbbd+BLq3JMAOhCKdBE3TcMGMZbKkDdJ5ivNdehOssMCIokNHFOsv7DO4rlEOfyKpg==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.24.0.tgz", + "integrity": "sha512-mjBaoo4ocxJppTorZVKWFpy1bfFj9FeCMJqzlMQGjpNPY9JwQi7OuS1axzNIk0nMX6jSgy6ZURDZ2w0QW6D56g==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.24.0.tgz", + "integrity": "sha512-ZXFk7M72R0YYFN5q13niV0B7G8/5dcQ9JDp8keJSfr3GoZeXEoMHP/HlvqROA3OMbMdfr19IjCeNAnPUG93b6A==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.24.0.tgz", + "integrity": "sha512-w1i+L7kAXZNdYl+vFvzSZy8Y1arS7vMgIy8wusXJzRrPyof5LAb02KGr1PD2EkRcl73kHulIID0M501lN+vobQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.24.0.tgz", + "integrity": "sha512-VXBrnPWgBpVDCVY6XF3LEW0pOU51KbaHhccHw6AS6vBWIC60eqsH19DAeeObl+g8nKAz04QFdl/Cefta0xQtUQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.24.0.tgz", + "integrity": "sha512-xrNcGDU0OxVcPTH/8n/ShH4UevZxKIO6HJFK0e15XItZP2UcaiLFd5kiX7hJnqCbSztUF8Qot+JWBC/QXRPYWQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.24.0.tgz", + "integrity": "sha512-fbMkAF7fufku0N2dE5TBXcNlg0pt0cJue4xBRE2Qc5Vqikxr4VCgKj/ht6SMdFcOacVA9rqF70APJ8RN/4vMJw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@swc/core": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core/-/core-1.7.26.tgz", + "integrity": "sha512-f5uYFf+TmMQyYIoxkn/evWhNGuUzC730dFwAKGwBVHHVoPyak1/GvJUm6i1SKl+2Hrj9oN0i3WSoWWZ4pgI8lw==", + "dev": true, + "hasInstallScript": true, + "dependencies": { + "@swc/counter": "^0.1.3", + "@swc/types": "^0.1.12" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/swc" + }, + "optionalDependencies": { + "@swc/core-darwin-arm64": "1.7.26", + "@swc/core-darwin-x64": "1.7.26", + "@swc/core-linux-arm-gnueabihf": "1.7.26", + "@swc/core-linux-arm64-gnu": "1.7.26", + "@swc/core-linux-arm64-musl": "1.7.26", + "@swc/core-linux-x64-gnu": "1.7.26", + "@swc/core-linux-x64-musl": "1.7.26", + "@swc/core-win32-arm64-msvc": "1.7.26", + "@swc/core-win32-ia32-msvc": "1.7.26", + "@swc/core-win32-x64-msvc": "1.7.26" + }, + "peerDependencies": { + "@swc/helpers": "*" + }, + "peerDependenciesMeta": { + "@swc/helpers": { + "optional": true + } + } + }, + "node_modules/@swc/core-darwin-arm64": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-darwin-arm64/-/core-darwin-arm64-1.7.26.tgz", + "integrity": "sha512-FF3CRYTg6a7ZVW4yT9mesxoVVZTrcSWtmZhxKCYJX9brH4CS/7PRPjAKNk6kzWgWuRoglP7hkjQcd6EpMcZEAw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-darwin-x64": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-darwin-x64/-/core-darwin-x64-1.7.26.tgz", + "integrity": "sha512-az3cibZdsay2HNKmc4bjf62QVukuiMRh5sfM5kHR/JMTrLyS6vSw7Ihs3UTkZjUxkLTT8ro54LI6sV6sUQUbLQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-arm-gnueabihf": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.7.26.tgz", + "integrity": "sha512-VYPFVJDO5zT5U3RpCdHE5v1gz4mmR8BfHecUZTmD2v1JeFY6fv9KArJUpjrHEEsjK/ucXkQFmJ0jaiWXmpOV9Q==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-arm64-gnu": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.7.26.tgz", + "integrity": "sha512-YKevOV7abpjcAzXrhsl+W48Z9mZvgoVs2eP5nY+uoMAdP2b3GxC0Df1Co0I90o2lkzO4jYBpTMcZlmUXLdXn+Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-arm64-musl": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.7.26.tgz", + "integrity": "sha512-3w8iZICMkQQON0uIcvz7+Q1MPOW6hJ4O5ETjA0LSP/tuKqx30hIniCGOgPDnv3UTMruLUnQbtBwVCZTBKR3Rkg==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-x64-gnu": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.7.26.tgz", + "integrity": "sha512-c+pp9Zkk2lqb06bNGkR2Looxrs7FtGDMA4/aHjZcCqATgp348hOKH5WPvNLBl+yPrISuWjbKDVn3NgAvfvpH4w==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-x64-musl": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.7.26.tgz", + "integrity": "sha512-PgtyfHBF6xG87dUSSdTJHwZ3/8vWZfNIXQV2GlwEpslrOkGqy+WaiiyE7Of7z9AvDILfBBBcJvJ/r8u980wAfQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-win32-arm64-msvc": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.7.26.tgz", + "integrity": "sha512-9TNXPIJqFynlAOrRD6tUQjMq7KApSklK3R/tXgIxc7Qx+lWu8hlDQ/kVPLpU7PWvMMwC/3hKBW+p5f+Tms1hmA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-win32-ia32-msvc": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.7.26.tgz", + "integrity": "sha512-9YngxNcG3177GYdsTum4V98Re+TlCeJEP4kEwEg9EagT5s3YejYdKwVAkAsJszzkXuyRDdnHUpYbTrPG6FiXrQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-win32-x64-msvc": { + "version": "1.7.26", + "resolved": "https://registry.npmjs.org/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.7.26.tgz", + "integrity": "sha512-VR+hzg9XqucgLjXxA13MtV5O3C0bK0ywtLIBw/+a+O+Oc6mxFWHtdUeXDbIi5AiPbn0fjgVJMqYnyjGyyX8u0w==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/counter": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@swc/counter/-/counter-0.1.3.tgz", + "integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==", + "dev": true + }, + "node_modules/@swc/types": { + "version": "0.1.12", + "resolved": "https://registry.npmjs.org/@swc/types/-/types-0.1.12.tgz", + "integrity": "sha512-wBJA+SdtkbFhHjTMYH+dEH1y4VpfGdAc2Kw/LK09i9bXd/K6j6PkDcFCEzb6iVfZMkPRrl/q0e3toqTAJdkIVA==", + "dev": true, + "dependencies": { + "@swc/counter": "^0.1.3" + } + }, + "node_modules/@types/estree": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz", + "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==", + "dev": true + }, + "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 + }, + "node_modules/@types/prop-types": { + "version": "15.7.13", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.13.tgz", + "integrity": "sha512-hCZTSvwbzWGvhqxp/RqVqwU999pBf2vp7hzIjiYOsl8wqOmUxkQ6ddw1cV3l8811+kdUFus/q4d1Y3E3SyEifA==", + "dev": true + }, + "node_modules/@types/react": { + "version": "18.3.11", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.11.tgz", + "integrity": "sha512-r6QZ069rFTjrEYgFdOck1gK7FLVsgJE7tTz0pQBczlBNUhBNk0MQH4UbnFSwjpQLMkLzgqvBBa+qGpLje16eTQ==", + "dev": true, + "dependencies": { + "@types/prop-types": "*", + "csstype": "^3.0.2" + } + }, + "node_modules/@types/react-dom": { + "version": "18.3.0", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.0.tgz", + "integrity": "sha512-EhwApuTmMBmXuFOikhQLIBUn6uFg81SwLMOAUgodJF14SOBOCMdU04gDoYi0WOJJHD144TL32z4yDqCW3dnkQg==", + "dev": true, + "dependencies": { + "@types/react": "*" + } + }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.8.0.tgz", + "integrity": "sha512-wORFWjU30B2WJ/aXBfOm1LX9v9nyt9D3jsSOxC3cCaTQGCW5k4jNpmjFv3U7p/7s4yvdjHzwtv2Sd2dOyhjS0A==", + "dev": true, + "dependencies": { + "@eslint-community/regexpp": "^4.10.0", + "@typescript-eslint/scope-manager": "8.8.0", + "@typescript-eslint/type-utils": "8.8.0", + "@typescript-eslint/utils": "8.8.0", + "@typescript-eslint/visitor-keys": "8.8.0", + "graphemer": "^1.4.0", + "ignore": "^5.3.1", + "natural-compare": "^1.4.0", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^8.0.0 || ^8.0.0-alpha.0", + "eslint": "^8.57.0 || ^9.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.8.0.tgz", + "integrity": "sha512-uEFUsgR+tl8GmzmLjRqz+VrDv4eoaMqMXW7ruXfgThaAShO9JTciKpEsB+TvnfFfbg5IpujgMXVV36gOJRLtZg==", + "dev": true, + "dependencies": { + "@typescript-eslint/scope-manager": "8.8.0", + "@typescript-eslint/types": "8.8.0", + "@typescript-eslint/typescript-estree": "8.8.0", + "@typescript-eslint/visitor-keys": "8.8.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.8.0.tgz", + "integrity": "sha512-EL8eaGC6gx3jDd8GwEFEV091210U97J0jeEHrAYvIYosmEGet4wJ+g0SYmLu+oRiAwbSA5AVrt6DxLHfdd+bUg==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "8.8.0", + "@typescript-eslint/visitor-keys": "8.8.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.8.0.tgz", + "integrity": "sha512-IKwJSS7bCqyCeG4NVGxnOP6lLT9Okc3Zj8hLO96bpMkJab+10HIfJbMouLrlpyOr3yrQ1cA413YPFiGd1mW9/Q==", + "dev": true, + "dependencies": { + "@typescript-eslint/typescript-estree": "8.8.0", + "@typescript-eslint/utils": "8.8.0", + "debug": "^4.3.4", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/types": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.8.0.tgz", + "integrity": "sha512-QJwc50hRCgBd/k12sTykOJbESe1RrzmX6COk8Y525C9l7oweZ+1lw9JiU56im7Amm8swlz00DRIlxMYLizr2Vw==", + "dev": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.8.0.tgz", + "integrity": "sha512-ZaMJwc/0ckLz5DaAZ+pNLmHv8AMVGtfWxZe/x2JVEkD5LnmhWiQMMcYT7IY7gkdJuzJ9P14fRy28lUrlDSWYdw==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "8.8.0", + "@typescript-eslint/visitor-keys": "8.8.0", + "debug": "^4.3.4", + "fast-glob": "^3.3.2", + "is-glob": "^4.0.3", + "minimatch": "^9.0.4", + "semver": "^7.6.0", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "dev": true, + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.8.0.tgz", + "integrity": "sha512-QE2MgfOTem00qrlPgyByaCHay9yb1+9BjnMFnSFkUKQfu7adBXDTnCAivURnuPPAG/qiB+kzKkZKmKfaMT0zVg==", + "dev": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@typescript-eslint/scope-manager": "8.8.0", + "@typescript-eslint/types": "8.8.0", + "@typescript-eslint/typescript-estree": "8.8.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.8.0.tgz", + "integrity": "sha512-8mq51Lx6Hpmd7HnA2fcHQo3YgfX1qbccxQOgZcb4tvasu//zXRaA1j5ZRFeCw/VRAdFi4mRM9DnZw0Nu0Q2d1g==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "8.8.0", + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/visitor-keys/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, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@vitejs/plugin-react-swc": { + "version": "3.7.1", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react-swc/-/plugin-react-swc-3.7.1.tgz", + "integrity": "sha512-vgWOY0i1EROUK0Ctg1hwhtC3SdcDjZcdit4Ups4aPkDcB1jYhmo+RMYWY87cmXMhvtD5uf8lV89j2w16vkdSVg==", + "dev": true, + "dependencies": { + "@swc/core": "^1.7.26" + }, + "peerDependencies": { + "vite": "^4 || ^5" + } + }, + "node_modules/acorn": { + "version": "8.12.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.1.tgz", + "integrity": "sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==", + "dev": true, + "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, + "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, + "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, + "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 + }, + "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 + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "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, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "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 + }, + "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 + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "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==", + "dev": true + }, + "node_modules/debug": { + "version": "4.3.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", + "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", + "dev": true, + "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 + }, + "node_modules/esbuild": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", + "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", + "dev": true, + "hasInstallScript": true, + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.21.5", + "@esbuild/android-arm": "0.21.5", + "@esbuild/android-arm64": "0.21.5", + "@esbuild/android-x64": "0.21.5", + "@esbuild/darwin-arm64": "0.21.5", + "@esbuild/darwin-x64": "0.21.5", + "@esbuild/freebsd-arm64": "0.21.5", + "@esbuild/freebsd-x64": "0.21.5", + "@esbuild/linux-arm": "0.21.5", + "@esbuild/linux-arm64": "0.21.5", + "@esbuild/linux-ia32": "0.21.5", + "@esbuild/linux-loong64": "0.21.5", + "@esbuild/linux-mips64el": "0.21.5", + "@esbuild/linux-ppc64": "0.21.5", + "@esbuild/linux-riscv64": "0.21.5", + "@esbuild/linux-s390x": "0.21.5", + "@esbuild/linux-x64": "0.21.5", + "@esbuild/netbsd-x64": "0.21.5", + "@esbuild/openbsd-x64": "0.21.5", + "@esbuild/sunos-x64": "0.21.5", + "@esbuild/win32-arm64": "0.21.5", + "@esbuild/win32-ia32": "0.21.5", + "@esbuild/win32-x64": "0.21.5" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "9.12.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.12.0.tgz", + "integrity": "sha512-UVIOlTEWxwIopRL1wgSQYdnVDcEvs2wyaO6DGo5mXqe3r16IoCNWkR29iHhyaP4cICWjbgbmFUGAhh0GJRuGZw==", + "dev": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.11.0", + "@eslint/config-array": "^0.18.0", + "@eslint/core": "^0.6.0", + "@eslint/eslintrc": "^3.1.0", + "@eslint/js": "9.12.0", + "@eslint/plugin-kit": "^0.2.0", + "@humanfs/node": "^0.16.5", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.3.1", + "@types/estree": "^1.0.6", + "@types/json-schema": "^7.0.15", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^8.1.0", + "eslint-visitor-keys": "^4.1.0", + "espree": "^10.2.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", + "text-table": "^0.2.0" + }, + "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.1.0-rc-fb9a90fa48-20240614", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-5.1.0-rc-fb9a90fa48-20240614.tgz", + "integrity": "sha512-xsiRwaDNF5wWNC4ZHLut+x/YcAxksUd9Rizt7LaEn3bV8VyYRpXnRJQlLOfYaVy9esk4DFP4zPPnoNVjq5Gc0w==", + "dev": true, + "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.12", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.12.tgz", + "integrity": "sha512-9neVjoGv20FwYtCP6CB1dzR1vr57ZDNOXst21wd2xJ/cTlM2xLq0GWVlSNTdMn/4BtP6cHYBMCSp1wFBJ9jBsg==", + "dev": true, + "peerDependencies": { + "eslint": ">=7" + } + }, + "node_modules/eslint-scope": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.1.0.tgz", + "integrity": "sha512-14dSvlhaVhKKsa9Fx1l8A17s7ah7Ef7wCakJ10LYk6+GYmP9yDti2oq2SEwcyndt6knfcZyhyxwY3i9yL78EQw==", + "dev": true, + "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.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.1.0.tgz", + "integrity": "sha512-Q7lok0mqMUSf5a/AdAZkA5a/gHcO6snwQClVNNvFKCAVlxXucdU8pKydU5ZVZjBx5xr37vGbFFWtLQYreLzrZg==", + "dev": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree": { + "version": "10.2.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.2.0.tgz", + "integrity": "sha512-upbkBJbckcCNBDBDXEbuhjbP68n+scUd3k/U2EkyM9nw+I/jPiL4cLF/Al06CF96wRltFda16sxDFrxsI1v0/g==", + "dev": true, + "dependencies": { + "acorn": "^8.12.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^4.1.0" + }, + "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, + "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, + "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, + "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, + "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 + }, + "node_modules/fast-glob": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", + "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.4" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/fast-glob/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "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 + }, + "node_modules/fastq": { + "version": "1.17.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", + "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "dev": true, + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/file-entry-cache": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", + "dev": true, + "dependencies": { + "flat-cache": "^4.0.0" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "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, + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.4" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/flatted": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", + "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", + "dev": true + }, + "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, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "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, + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "15.10.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-15.10.0.tgz", + "integrity": "sha512-tqFIbz83w4Y5TCbtgjZjApohbuh7K9BxGYFm7ifwDR240tvdb7P9x+/9VvUKlmkPoiknoJtanI8UOrqxS3a7lQ==", + "dev": true, + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "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, + "engines": { + "node": ">=8" + } + }, + "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, + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "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, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "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, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "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==" + }, + "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, + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "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 + }, + "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 + }, + "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 + }, + "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, + "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, + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "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 + }, + "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==", + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.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, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/nanoid": { + "version": "3.3.7", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", + "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "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 + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "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, + "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, + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "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, + "engines": { + "node": ">=8" + } + }, + "node_modules/picocolors": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.0.tgz", + "integrity": "sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==", + "dev": true + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/postcss": { + "version": "8.4.47", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.47.tgz", + "integrity": "sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==", + "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" + } + ], + "dependencies": { + "nanoid": "^3.3.7", + "picocolors": "^1.1.0", + "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, + "engines": { + "node": ">= 0.8.0" + } + }, + "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, + "engines": { + "node": ">=6" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/react": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz", + "integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==", + "dependencies": { + "loose-envify": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz", + "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==", + "dependencies": { + "loose-envify": "^1.1.0", + "scheduler": "^0.23.2" + }, + "peerDependencies": { + "react": "^18.3.1" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rollup": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.24.0.tgz", + "integrity": "sha512-DOmrlGSXNk1DM0ljiQA+i+o0rSLhtii1je5wgk60j49d1jHT5YYttBv1iWOnYSTG+fZZESUOSNiAl89SIet+Cg==", + "dev": true, + "dependencies": { + "@types/estree": "1.0.6" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.24.0", + "@rollup/rollup-android-arm64": "4.24.0", + "@rollup/rollup-darwin-arm64": "4.24.0", + "@rollup/rollup-darwin-x64": "4.24.0", + "@rollup/rollup-linux-arm-gnueabihf": "4.24.0", + "@rollup/rollup-linux-arm-musleabihf": "4.24.0", + "@rollup/rollup-linux-arm64-gnu": "4.24.0", + "@rollup/rollup-linux-arm64-musl": "4.24.0", + "@rollup/rollup-linux-powerpc64le-gnu": "4.24.0", + "@rollup/rollup-linux-riscv64-gnu": "4.24.0", + "@rollup/rollup-linux-s390x-gnu": "4.24.0", + "@rollup/rollup-linux-x64-gnu": "4.24.0", + "@rollup/rollup-linux-x64-musl": "4.24.0", + "@rollup/rollup-win32-arm64-msvc": "4.24.0", + "@rollup/rollup-win32-ia32-msvc": "4.24.0", + "@rollup/rollup-win32-x64-msvc": "4.24.0", + "fsevents": "~2.3.2" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/scheduler": { + "version": "0.23.2", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz", + "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==", + "dependencies": { + "loose-envify": "^1.1.0" + } + }, + "node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "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, + "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, + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map-js": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", + "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", + "dev": true, + "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, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/ts-api-utils": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.3.0.tgz", + "integrity": "sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ==", + "dev": true, + "engines": { + "node": ">=16" + }, + "peerDependencies": { + "typescript": ">=4.2.0" + } + }, + "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, + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/typescript": { + "version": "5.6.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.6.2.tgz", + "integrity": "sha512-NW8ByodCSNCwZeghjN3o+JX5OFH0Ojg6sadjEKY4huZ52TqbJTJnDo5+Tw98lSy63NZvi4n+ez5m2u5d4PkZyw==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/typescript-eslint": { + "version": "8.8.0", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.8.0.tgz", + "integrity": "sha512-BjIT/VwJ8+0rVO01ZQ2ZVnjE1svFBiRczcpr1t1Yxt7sT25VSbPfrJtDsQ8uQTy2pilX5nI9gwxhUyLULNentw==", + "dev": true, + "dependencies": { + "@typescript-eslint/eslint-plugin": "8.8.0", + "@typescript-eslint/parser": "8.8.0", + "@typescript-eslint/utils": "8.8.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "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, + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/vite": { + "version": "5.4.8", + "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.8.tgz", + "integrity": "sha512-FqrItQ4DT1NC4zCUqMB4c4AZORMKIa0m8/URVCZ77OZ/QSNeJ54bU1vrFADbDsuwfIPcgknRkmqakQcgnL4GiQ==", + "dev": true, + "dependencies": { + "esbuild": "^0.21.3", + "postcss": "^8.4.43", + "rollup": "^4.20.0" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^18.0.0 || >=20.0.0", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "sass-embedded": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.4.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "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, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + } +} diff --git a/frontend/src/App.css b/frontend/src/app.css similarity index 100% rename from frontend/src/App.css rename to frontend/src/app.css diff --git a/frontend/src/App.tsx b/frontend/src/app.tsx similarity index 97% rename from frontend/src/App.tsx rename to frontend/src/app.tsx index afe48ac..da6781a 100644 --- a/frontend/src/App.tsx +++ b/frontend/src/app.tsx @@ -1,7 +1,7 @@ import { useState } from 'react' import reactLogo from './assets/react.svg' import viteLogo from '/vite.svg' -import './App.css' +import './app.css' function App() { const [count, setCount] = useState(0) diff --git a/frontend/src/main.tsx b/frontend/src/main.tsx index 6f4ac9b..67b1c49 100644 --- a/frontend/src/main.tsx +++ b/frontend/src/main.tsx @@ -1,6 +1,6 @@ import { StrictMode } from 'react' import { createRoot } from 'react-dom/client' -import App from './App.tsx' +import App from './app.tsx' import './index.css' createRoot(document.getElementById('root')!).render( From a6b16a1abe89768f7c008ddbb1091943db422194 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 5 Oct 2024 13:30:46 -0400 Subject: [PATCH 15/27] Continue on frontend --- frontend/index.html | 23 ++++--- frontend/package-lock.json | 91 +++++++++++++++++++++++++++ frontend/package.json | 1 + frontend/public/favicon.ico | Bin 0 -> 2933 bytes frontend/src/app.tsx | 27 +------- frontend/src/data/command.ts | 14 +++++ frontend/src/data/main_command.ts | 8 +++ frontend/src/environment.ts | 1 + frontend/src/input/command_input.tsx | 17 +++++ frontend/src/input/input_api.ts | 12 ++++ 10 files changed, 159 insertions(+), 35 deletions(-) create mode 100644 frontend/public/favicon.ico create mode 100644 frontend/src/data/command.ts create mode 100644 frontend/src/data/main_command.ts create mode 100644 frontend/src/environment.ts create mode 100644 frontend/src/input/command_input.tsx create mode 100644 frontend/src/input/input_api.ts diff --git a/frontend/index.html b/frontend/index.html index e4b78ea..54edcdc 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -1,13 +1,16 @@ - - - - - Vite + React + TS - - -
- - + + + + + + GS Onboarding + + + +
+ + + diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 4877c0a..22e90e0 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -8,6 +8,7 @@ "name": "frontend", "version": "0.0.0", "dependencies": { + "axios": "^1.7.7", "react": "^18.3.1", "react-dom": "^18.3.1" }, @@ -1346,6 +1347,21 @@ "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", "dev": true }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" + }, + "node_modules/axios": { + "version": "1.7.7", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.7.tgz", + "integrity": "sha512-S4kL7XrjgBmvdGut0sN3yJxqYzrDOnivkBiN0OFs6hLiUam3UPvswUo0kqGyhqUZGEOytHyumEdXsAkgCOUf3Q==", + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, "node_modules/balanced-match": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", @@ -1417,6 +1433,17 @@ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, + "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==", + "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", @@ -1466,6 +1493,14 @@ "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", "dev": true }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/esbuild": { "version": "0.21.5", "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", @@ -1798,6 +1833,38 @@ "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", "dev": true }, + "node_modules/follow-redirects": { + "version": "1.15.9", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", + "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/form-data": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", + "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "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", @@ -2032,6 +2099,25 @@ "node": ">=8.6" } }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "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==", + "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", @@ -2206,6 +2292,11 @@ "node": ">= 0.8.0" } }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, "node_modules/punycode": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", diff --git a/frontend/package.json b/frontend/package.json index f5a179a..2e155c8 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -10,6 +10,7 @@ "preview": "vite preview" }, "dependencies": { + "axios": "^1.7.7", "react": "^18.3.1", "react-dom": "^18.3.1" }, diff --git a/frontend/public/favicon.ico b/frontend/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..4ef6ad0f80a6268efd8797de1d981f85f6a3c109 GIT binary patch literal 2933 zcmV-*3ySoKP)R9JX)&_q+FgcX@<7<^cfmdf##W&Bpn2X9Tu>=ZC+O^Jh;J#be$H^7SIG z_Wq9V$4BzR_uosM@89U}cdq~XW3gCVLYa_HZ{6aDT)BLi|8KWiBX3^sUv}bHZ_mLO z_nz+B*?OgGXWKh3?&<2=w|7tLdsnW^!u^3xCGf%bspQAIcgdjx`+gmS-EJqhzy8_< zAUQ5p7?=y+eRI141^s@*>Q(lv)D&2~vIdq^&WEabbD?F+W>{ZW3uT3QFuS-AigL4l zdaiB9%P2VB3FPeQlR|(|KVV(@OX2tK-6P+9`>j*Bgqg7R>7{HwW#QcN{``zI@YAc{ zh37kAVR@M~BRSDJV`{u@R$-p497omM*|w^Ra!YsTF58+F%OMZ^U=*4%L}tJAZ=;_LzESxh7I@`8eV<(ISZ@mVjC`heVQ_)}qJq}Sc;1EGH7{Fa7W9X^eO`zUd1(fDx z(X*#cS`uTPgc-?6HatJ&M*lT};Dn`B3(4_UUUta5@a8!VBtCkQP&v1p{PEVuE2QF8fUv32WTMeAF`&_ELE+)X*0^v33YVkssHiB2h=>4%LIGN>mRn_B zSut#?Uu{+5UbM*1C~zDT`p%_`4t`PUu)xh5*QGz-|2Lb*%`C`eo1v~|shz#3*Xx_671?C+26HF*w z&esh_6X^7MKA1aPmXRxHSg4sILBU7uVHt`FU>?~vadbEd85+Q?K-yZi3Ef?t1S!{mc&H(vx(SHJ3&JuNz}((1AhYvbNN+k0!SUIkRLDW2Qi7)- z0hvHREteqxJzBuPTmkML9#kn1IwmB@?yK>Hy!6z6q2M5{wB0Uov9WdQmiHO)>Xt9F zY1}2;@4*Q74^bx1vSjzJc8r9xD_FR!s##{yqUPqz zN?Oo<28ienv9;Y$a`Y=$c=8S;Ep6e#8Hl*WO;EV&Ed-GQL?UoOY@KCauh8+3v#k$O z*6qXpB%sl#K}0f#1p3nwv~p(ZbPEa}{l#DZ;>ZR2diEShiI0WWrba6hFumb0RP=rc zhLPhy<`oDfd#=FH*i1akos(|jtX04P4p>63cQBu`ABpHBZ5CRES%ycBUZjEEM!I52 zm5pV}OBc@1JbL&g2Y{ZgU2lu%jfvqQHnqu@=50R-!(!6FQ>g*DrziSb1}r2a)gIvC z8w4|Zu41(k!P8C3Bfu%#HH=2|y_Q?0sOusGMW=EQURo8MUshsa85Yxe?Z$O=4gi>) zt{dD4j2aeTGkN<^L8f$Qh6G~aWI(OdCT|!oC9q(@YKU6813Z~;Q8?3hAnqT|@!MeZ z;zTs3&4H|E4{-qAqK>A-PqrjYj)HwX-Hl1HQ4Ro@Zm;QZLqzZ}n?bLmJelxRD{?cd zR4T_D{kdlL5|}bJ3g-k&2o6o_&>Ds6`EUZkJb)QfH!Kvnavm6|<;brk3o0!0XU&Ap z_SQ|wQ(_zd_U`UH=ciM^gmAOXB0g`3S6tj)sv4z)dHD zQPjIM7eyxs$8U{F0UiclDA|1(0l0yeMn)IRDz&6fPlA1WyKA;?e%4W$dXF4_zN{b* z5~C(sMYY1E6%%SUn;j{3fLv(}FTu#TY^d>;!#JZJh)T^thy&V-EMF+pQQAjBEeK;#NX`V}9#fN_BB56hg7`C&hs)3Kq!bY^}w6lG_y zY&e4Ffq^<;F3!!mzrJoIM24HGTB)E;zdmHQ0aTtcQ27KxSvN+3+Qf50U8p~l4?%Z% z$UuQb$;g0U6lgK_Srinb`A0sPOj#7o%+Il5D#i-A=JMcvg!ceDTeh8IR$v{EjF(yi zbuwD1R1VaAM>QN*xD1N8Sp`nUY@-S`kMe~O%=JW}bfx0#&A)}THG8@C=Ab}|<=vVb zHyKWy=&NQOS6lN|f&BTiPlelm`-=SQr+aDG2SRh>Cfme_Q4|@>qJc91CnO|*C*}iZ zrDGtXE7o)EMoh`VvaEyvT&p(8p+xTiLBq_D-gX9NY{dFOB$O#soReW^tsN$zuhHT< zW_i2aCV8Y2+TGE9f>~i#YqLd*i3v$ajaZU>eS8KnU37sMg2gAh@hB8xR!rS=1d=x% zg1RjSVdv~cuqiVi{7`c@se~p*Pox`G*IHPW?SJoTReQ@eGN-IW7|a1@4ac2pU$`S|dV&!5y{G z@yIZ^EWm30kibCj_45NHEfUn-5q8+XsFX@-Lf4HN8HCx@{{icUD#}VfN1?{S3bXqH zN7A@)Jx3KgAyUAep4`22r)YX&97K;B4UHStStgGkOGTwSZ%SFe$J#AV49_RA4-5+r z2RY^owub#8@18=24S`7W2z!92vUn>5z$Wx_XA{zQ?3nQGrHka$>#se&Gu8=2*weX- z^$AJkC57LZ@vs@w5@FTTODyHZ`F2*(M5T<%q}=?xYf`yMsEB&jh_h&;+S_%|o&VZtv zOvp`70|fe@zHa5UqlXW!U|M&yZg-w5y>R|KVZZ<9C6%>kzv<~FRx7`#O4R&<=>np_ zMl$xbS6`{Zi)kZ{j?VTSZJj&Y>d&2igI!F+*%_HB$y`uDR)+I_Nlw1j_y66HotF6I z*2gZ3{B-X}sb&DkhacP^-CfU-)TB6G$W%Dc6$Gmrx%1C|@cl)(*}{VdKTFtrLPQu} fyLRo`V`cpZ5dWBKd3HYb00000NkvXXu0mjfei5CB literal 0 HcmV?d00001 diff --git a/frontend/src/app.tsx b/frontend/src/app.tsx index da6781a..c383bbe 100644 --- a/frontend/src/app.tsx +++ b/frontend/src/app.tsx @@ -1,33 +1,10 @@ -import { useState } from 'react' -import reactLogo from './assets/react.svg' -import viteLogo from '/vite.svg' import './app.css' +import CommandInput from './input/command_input' function App() { - const [count, setCount] = useState(0) - return ( <> - -

Vite + React

-
- -

- Edit src/App.tsx and save to test HMR -

-
-

- Click on the Vite and React logos to learn more -

+ ) } diff --git a/frontend/src/data/command.ts b/frontend/src/data/command.ts new file mode 100644 index 0000000..21fb03e --- /dev/null +++ b/frontend/src/data/command.ts @@ -0,0 +1,14 @@ +export interface CommandRequest { + name: string + params: string | null + format: string | null +} + +export interface CommandResponse { + id: number + name: string + params: string | null + format: string | null + created_on: string + updated_on: string +} diff --git a/frontend/src/data/main_command.ts b/frontend/src/data/main_command.ts new file mode 100644 index 0000000..c8d4096 --- /dev/null +++ b/frontend/src/data/main_command.ts @@ -0,0 +1,8 @@ +export interface MainCommandResponse { + id: number; + name: string + params: string | null; + format: string | null; + data_size: number; + total_size: number; +} diff --git a/frontend/src/environment.ts b/frontend/src/environment.ts new file mode 100644 index 0000000..7a0eff9 --- /dev/null +++ b/frontend/src/environment.ts @@ -0,0 +1 @@ +export const API_URL = "http://localhost:8000" diff --git a/frontend/src/input/command_input.tsx b/frontend/src/input/command_input.tsx new file mode 100644 index 0000000..5632d18 --- /dev/null +++ b/frontend/src/input/command_input.tsx @@ -0,0 +1,17 @@ +const CommandInput = () => { + return ( + <> +
+ + + +
+ + ) +} + +export default CommandInput; diff --git a/frontend/src/input/input_api.ts b/frontend/src/input/input_api.ts new file mode 100644 index 0000000..391ad6c --- /dev/null +++ b/frontend/src/input/input_api.ts @@ -0,0 +1,12 @@ +import { API_URL } from "../environment"; +import { CommandRequest, CommandResponse } from "../data/command"; +import axios from "axios"; + +export const createCommand = async (requestData: CommandRequest): Promise => { + try { + const { data } = await axios.post(`${API_URL}/commands`, requestData); + return data + } catch (error) { + console.error(error); + } +} From 943782a56360e634177b22d386307cc85c7a5f90 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 5 Oct 2024 15:07:04 -0400 Subject: [PATCH 16/27] Move endpoints around, add main-command get --- backend/api/middlewares/cors_middleware.py | 2 +- backend/api/models/response_model.py | 10 +++++++++- .../{endpoints.py => resources/command.py} | 10 +++++----- backend/api/resources/main_command.py | 20 +++++++++++++++++++ backend/main.py | 6 ++++-- 5 files changed, 39 insertions(+), 9 deletions(-) rename backend/api/{endpoints.py => resources/command.py} (81%) create mode 100644 backend/api/resources/main_command.py diff --git a/backend/api/middlewares/cors_middleware.py b/backend/api/middlewares/cors_middleware.py index 11f14f7..e31581a 100644 --- a/backend/api/middlewares/cors_middleware.py +++ b/backend/api/middlewares/cors_middleware.py @@ -10,7 +10,7 @@ def add_cors_middleware(app: FastAPI) -> None: """ app.add_middleware( CORSMiddleware, - allow_origins=["http://localhost:*"], + allow_origins=["http://localhost:5173"], allow_credentials=True, allow_methods=["*"], allow_headers=["*"], diff --git a/backend/api/models/response_model.py b/backend/api/models/response_model.py index 6befa14..d541eb8 100644 --- a/backend/api/models/response_model.py +++ b/backend/api/models/response_model.py @@ -1,6 +1,6 @@ from pydantic import BaseModel -from backend.data.data_models import Command +from backend.data.data_models import Command, MainCommand class CommandListResponse(BaseModel): @@ -9,3 +9,11 @@ class CommandListResponse(BaseModel): """ data: list[Command] + + +class MainCommandListResponse(BaseModel): + """ + List of main commands + """ + + data: list[MainCommand] diff --git a/backend/api/endpoints.py b/backend/api/resources/command.py similarity index 81% rename from backend/api/endpoints.py rename to backend/api/resources/command.py index 7de7c01..b675ad3 100644 --- a/backend/api/endpoints.py +++ b/backend/api/resources/command.py @@ -6,15 +6,15 @@ from backend.data.data_models import Command from backend.data.engine import get_db -resource = APIRouter() +command_router = APIRouter(tags=["Commands"]) -@resource.get("/", response_model=CommandListResponse) +@command_router.get("/", response_model=CommandListResponse) async def get_items(db: Session = Depends(get_db)): """ Gets all the items - @return Returns a list of items + @return Returns a list of commands """ query = select(Command) items = db.exec(query).all() @@ -22,7 +22,7 @@ async def get_items(db: Session = Depends(get_db)): return {"data": items} -@resource.post("/", response_model=Command) +@command_router.post("/", response_model=Command) async def create_item(payload: CommandRequest): """ Creates an item with the given payload and returns the payload with some other information @@ -33,7 +33,7 @@ async def create_item(payload: CommandRequest): # TODO: Implement this endpoint -@resource.delete("/{id}") +@command_router.delete("/{id}") async def delete_item(id: int): """ Deletes the item with the given id if it exists. Otherwise raises a 404 error. diff --git a/backend/api/resources/main_command.py b/backend/api/resources/main_command.py new file mode 100644 index 0000000..ae03be2 --- /dev/null +++ b/backend/api/resources/main_command.py @@ -0,0 +1,20 @@ +from fastapi import APIRouter, Depends +from sqlmodel import Session, select + +from backend.api.models.response_model import MainCommandListResponse +from backend.data.data_models import MainCommand +from backend.data.engine import get_db + +main_command_router = APIRouter(tags=["Main Commands"]) + + +@main_command_router.get("/", response_model=MainCommandListResponse) +async def get_main_commands(db: Session = Depends(get_db)): + """ + Gets all the main commands that can be created. + + @return Returns a list of main commands + """ + query = select(MainCommand) + items = db.exec(query).all() + return {"data": items} diff --git a/backend/main.py b/backend/main.py index 82e04b9..6fcd9e8 100644 --- a/backend/main.py +++ b/backend/main.py @@ -1,4 +1,5 @@ -from backend.api.endpoints import resource +from backend.api.resources.command import command_router +from backend.api.resources.main_command import main_command_router from backend.api.lifespan import lifespan from backend.api.middlewares.cors_middleware import add_cors_middleware from backend.api.middlewares.logger_middleware import LoggerMiddleware @@ -7,7 +8,8 @@ def setup_routes(app: FastAPI) -> None: """Adds the routes to the app""" - app.include_router(router=resource) + app.include_router(router=command_router, prefix="/commands") + app.include_router(router=main_command_router, prefix="/main-commands") def setup_middlewares(app: FastAPI) -> None: From a251cf0d0735c3afd055fe5ac7f179b7f9ce82cf Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 5 Oct 2024 15:16:13 -0400 Subject: [PATCH 17/27] Setup basic input form and seperate requests/responses --- frontend/src/data/command.ts | 14 -------------- frontend/src/data/main_command.ts | 8 -------- frontend/src/data/request.ts | 6 ++++++ frontend/src/data/response.ts | 21 +++++++++++++++++++++ frontend/src/input/command_input.css | 7 +++++++ frontend/src/input/command_input.tsx | 27 ++++++++++++++++++++------- frontend/src/input/input_api.ts | 15 ++++++++++++++- 7 files changed, 68 insertions(+), 30 deletions(-) delete mode 100644 frontend/src/data/command.ts delete mode 100644 frontend/src/data/main_command.ts create mode 100644 frontend/src/data/request.ts create mode 100644 frontend/src/data/response.ts create mode 100644 frontend/src/input/command_input.css diff --git a/frontend/src/data/command.ts b/frontend/src/data/command.ts deleted file mode 100644 index 21fb03e..0000000 --- a/frontend/src/data/command.ts +++ /dev/null @@ -1,14 +0,0 @@ -export interface CommandRequest { - name: string - params: string | null - format: string | null -} - -export interface CommandResponse { - id: number - name: string - params: string | null - format: string | null - created_on: string - updated_on: string -} diff --git a/frontend/src/data/main_command.ts b/frontend/src/data/main_command.ts deleted file mode 100644 index c8d4096..0000000 --- a/frontend/src/data/main_command.ts +++ /dev/null @@ -1,8 +0,0 @@ -export interface MainCommandResponse { - id: number; - name: string - params: string | null; - format: string | null; - data_size: number; - total_size: number; -} diff --git a/frontend/src/data/request.ts b/frontend/src/data/request.ts new file mode 100644 index 0000000..d069f36 --- /dev/null +++ b/frontend/src/data/request.ts @@ -0,0 +1,6 @@ +export interface CommandRequest { + name: string + params: string | null + format: string | null +} + diff --git a/frontend/src/data/response.ts b/frontend/src/data/response.ts new file mode 100644 index 0000000..7289d56 --- /dev/null +++ b/frontend/src/data/response.ts @@ -0,0 +1,21 @@ +export interface MainCommandResponse { + id: number; + name: string + params: string | null; + format: string | null; + data_size: number; + total_size: number; +} + +export interface MainCommandListResponse { + data: MainCommandResponse[] +} + +export interface CommandResponse { + id: number + name: string + params: string | null + format: string | null + created_on: string + updated_on: string +} diff --git a/frontend/src/input/command_input.css b/frontend/src/input/command_input.css new file mode 100644 index 0000000..bf92573 --- /dev/null +++ b/frontend/src/input/command_input.css @@ -0,0 +1,7 @@ +.spreader { + display: flex; + flex-direction: column; + justify-content: center; + align-items: center; + gap: 4px; +} diff --git a/frontend/src/input/command_input.tsx b/frontend/src/input/command_input.tsx index 5632d18..2ca901a 100644 --- a/frontend/src/input/command_input.tsx +++ b/frontend/src/input/command_input.tsx @@ -1,14 +1,27 @@ +import "./command_input.css" + const CommandInput = () => { + // Setup state and useEffect calls here + + const handleSubmit = () => { + // TODO:(Member) Submit to your post endpoint + } + return ( <>
- - - +
+
+ + +
+ {/*Add input handling here if the selected command has a param input*/} + +
) diff --git a/frontend/src/input/input_api.ts b/frontend/src/input/input_api.ts index 391ad6c..e6654e2 100644 --- a/frontend/src/input/input_api.ts +++ b/frontend/src/input/input_api.ts @@ -1,6 +1,7 @@ import { API_URL } from "../environment"; -import { CommandRequest, CommandResponse } from "../data/command"; +import { CommandRequest } from "../data/request"; import axios from "axios"; +import { CommandResponse, MainCommandListResponse } from "../data/response"; export const createCommand = async (requestData: CommandRequest): Promise => { try { @@ -8,5 +9,17 @@ export const createCommand = async (requestData: CommandRequest): Promise => { + try { + const { data } = await axios.get(`${API_URL}/main-commands/`); + console.log(data) + return data; + } catch (error) { + console.error(error) + throw error; } } From db19eb4a21eb131c482424c3dcb5097508c0b467 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sat, 5 Oct 2024 15:47:58 -0400 Subject: [PATCH 18/27] Create table --- frontend/src/app.tsx | 3 +++ frontend/src/data/response.ts | 4 +++ frontend/src/display/command_api.ts | 13 +++++++++ frontend/src/display/row.tsx | 21 +++++++++++++++ frontend/src/display/table.tsx | 42 +++++++++++++++++++++++++++++ 5 files changed, 83 insertions(+) create mode 100644 frontend/src/display/command_api.ts create mode 100644 frontend/src/display/row.tsx create mode 100644 frontend/src/display/table.tsx diff --git a/frontend/src/app.tsx b/frontend/src/app.tsx index c383bbe..aaa5946 100644 --- a/frontend/src/app.tsx +++ b/frontend/src/app.tsx @@ -1,10 +1,13 @@ import './app.css' +import CommandTable from './display/table' import CommandInput from './input/command_input' function App() { return ( <> +

Command List:

+ ) } diff --git a/frontend/src/data/response.ts b/frontend/src/data/response.ts index 7289d56..5d9003d 100644 --- a/frontend/src/data/response.ts +++ b/frontend/src/data/response.ts @@ -19,3 +19,7 @@ export interface CommandResponse { created_on: string updated_on: string } + +export interface CommandListReponse { + data: CommandResponse[] +} diff --git a/frontend/src/display/command_api.ts b/frontend/src/display/command_api.ts new file mode 100644 index 0000000..f371bff --- /dev/null +++ b/frontend/src/display/command_api.ts @@ -0,0 +1,13 @@ +import axios from "axios"; +import { CommandListReponse } from "../data/response"; +import { API_URL } from "../environment"; + +export const getCommands = async (): Promise => { + try { + const { data } = await axios.get(`${API_URL}/commands/`) + return data; + } catch (error) { + console.error(error) + throw error + } +} diff --git a/frontend/src/display/row.tsx b/frontend/src/display/row.tsx new file mode 100644 index 0000000..569221e --- /dev/null +++ b/frontend/src/display/row.tsx @@ -0,0 +1,21 @@ +import { CommandResponse } from "../data/response"; + +interface CommandRowProp extends CommandResponse { + handleDelete: () => void; +} + +const CommandRow = ({ id, name, params, format, created_on, updated_on, handleDelete }: CommandRowProp) => { + return ( + + {id} + {name} + {params} + {format} + {created_on} + {updated_on} + + + ) +} + +export default CommandRow; diff --git a/frontend/src/display/table.tsx b/frontend/src/display/table.tsx new file mode 100644 index 0000000..7a895bd --- /dev/null +++ b/frontend/src/display/table.tsx @@ -0,0 +1,42 @@ +import { useEffect, useState } from "react" +import { CommandResponse } from "../data/response" +import { getCommands } from "./command_api" +import CommandRow from "./row" + +const CommandTable = () => { + const [commands, setCommands] = useState([]) + + const getCommandsFn = async () => { + const data = await getCommands(); + setCommands(data.data) + } + + useEffect(() => { + getCommandsFn(); + }, []) + + const handleDelete = (id: number) => { + return () => { setCommands(commands.filter((value) => value.id != id)) } + } + + return ( + + + + + + + + + + + + + + {commands.map(value => ())} + +
ID: Name: Params: Format: Created On: Updated On: Delete
+ ) +} + +export default CommandTable; From 54ff6353d52745e638a0017d4c5894332acd11b9 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 6 Oct 2024 12:35:25 -0400 Subject: [PATCH 19/27] Add mock data to backend --- .gitignore | 2 ++ backend/api/lifespan.py | 23 +++++++++++++--- backend/data/data_models.py | 8 ++++-- backend/data/engine.py | 2 +- backend/data/mock_data.py | 29 ++++++++++++++++++++ backend/utils/time.py | 5 ++++ db.sql | Bin 12288 -> 0 bytes test/backend/conftest.py | 51 +++++++----------------------------- test/backend/test_setup.py | 6 +++-- 9 files changed, 75 insertions(+), 51 deletions(-) create mode 100644 backend/data/mock_data.py create mode 100644 backend/utils/time.py delete mode 100644 db.sql diff --git a/.gitignore b/.gitignore index 7949c3c..2174718 100644 --- a/.gitignore +++ b/.gitignore @@ -183,3 +183,5 @@ lerna-debug.log* *dist-ssr *.local +# Db +*.db diff --git a/backend/api/lifespan.py b/backend/api/lifespan.py index addab1f..13c0ce6 100644 --- a/backend/api/lifespan.py +++ b/backend/api/lifespan.py @@ -1,13 +1,28 @@ from contextlib import asynccontextmanager +from datetime import datetime from fastapi import FastAPI -from sqlmodel import SQLModel -import backend.data.data_models +from sqlmodel import SQLModel, select +from backend.data.data_models import MainCommand from backend.data.engine import get_db +from backend.data.mock_data import commands, main_commands +from backend.utils.time import to_unix_time @asynccontextmanager async def lifespan(_: FastAPI): - print("Starting the app") + print("Starting application") SQLModel.metadata.create_all(get_db().connection()) + default_time = "2024-01-01T00:00:00" + default_datetime = datetime.strptime(default_time, "%Y-%m-%dT%H:%M:%S") + unix_time = to_unix_time(default_datetime) + # Setup the db with mock data + with get_db() as session: + query = select(MainCommand).limit(1) # Check if the db is empty + result = session.exec(query).first() + print(result) + if result is None: + session.add_all(main_commands()) + session.commit() + session.add_all(commands(unix_time)) + session.commit() yield - print("Stopping the app") diff --git a/backend/data/data_models.py b/backend/data/data_models.py index 0624062..777f6eb 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -15,7 +15,9 @@ class MainCommand(SQLModel, table=True): List of commands: https://docs.google.com/spreadsheets/d/1XWXgp3--NHZ4XlxOyBYPS-M_LOU_ai-I6TcvotKhR1s/edit?gid=564815068#gid=564815068 """ - id: int | None = Field(default=None, primary_key=True) + id: int | None = Field( + default=None, primary_key=True + ) # NOTE: Must be None for autoincrement name: str params: str | None = None format: str | None = None @@ -44,7 +46,9 @@ class Command(SQLModel, table=True): This table holds the data related to actual commands sent from the ground station up to the OBC. """ - id: int | None = Field(primary_key=True) + id: int | None = Field( + default=None, primary_key=True + ) # NOTE: Must be None for autoincrement command_type: int = Field( foreign_key="maincommand.id" ) # Forign key must be a string diff --git a/backend/data/engine.py b/backend/data/engine.py index 519fb49..54d3c69 100644 --- a/backend/data/engine.py +++ b/backend/data/engine.py @@ -1,7 +1,7 @@ from typing import Final from sqlmodel import Session, create_engine -SQL_PATH: Final[str] = "sqlite:///db.sql" +SQL_PATH: Final[str] = "sqlite:///sqlite.db" def get_db() -> Session: diff --git a/backend/data/mock_data.py b/backend/data/mock_data.py new file mode 100644 index 0000000..dfeaa5a --- /dev/null +++ b/backend/data/mock_data.py @@ -0,0 +1,29 @@ +from backend.data.data_models import Command, MainCommand + + +def commands(unix_time: int) -> list[Command]: + return [ + Command(command_type=1, params=f"{unix_time}"), # id=1, RTC Sync for 2021-01-01 + Command( + command_type=2, params=f"1,{unix_time}" + ), # id=2, Emergency mode for 2021-01-01 + ] + + +def main_commands() -> list[MainCommand]: + return [ + MainCommand( + name="RTC Sync", + params="time", + format="int 7 bytes", + data_size=7, + total_size=8, + ), # Will have id of 1 + MainCommand( + name="Manually activate an emergency mode for a specified amount of time", + params="mode_state_number,time", + format="int 1 byte, int 7 bytes", + data_size=8, + total_size=9, + ), # Will have id of 2 + ] diff --git a/backend/utils/time.py b/backend/utils/time.py new file mode 100644 index 0000000..d90c147 --- /dev/null +++ b/backend/utils/time.py @@ -0,0 +1,5 @@ +from datetime import datetime + + +def to_unix_time(time: datetime) -> int: + return int(time.timestamp()) diff --git a/db.sql b/db.sql deleted file mode 100644 index 585abf290650ba6b9d05ea41cfdd8d9ac6d4be62..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 12288 zcmeI#O;5rw7zglfm=G{TZydNh35nSfxx*iT?b z1{n$*J&=Est}jp9^!e@7`#U#MA-yD%SSmVYhs-e8IT2&b)U}}NqNL-!#Xz@4uD_{k zvXiIlijJCy!83y2n&5Wh?sXbsVl;FjZ2`ru68D#+4(i zGkazGSWaZTjLr|7gi?h;KN-=bzKGxM@Rc32aj-RhE7~1L^yr57$ch3xJ+s*ryxsEB zIV*o*n*?ui!Mz5*r&va#Wxjf2*?u1w#cI87eD2Soatm@@a|J9H*n&2aakvm!N$xL4 z_Abx((qwe{vOkU9)^n>w$>HYkZ=JH8zRmTc9|ZysfB*y_009U<00Izz00bZafgKjW e`+tXjE>eR41Rwwb2tWV=5P$##AOHafWCB0#5TOA8 diff --git a/test/backend/conftest.py b/test/backend/conftest.py index 14cc8dd..420b071 100644 --- a/test/backend/conftest.py +++ b/test/backend/conftest.py @@ -1,9 +1,8 @@ -from _pytest import scope +from datetime import datetime from sqlmodel import SQLModel, Session, create_engine import pytest -from datetime import datetime - -from backend.data.data_models import MainCommand, Command +from backend.data.mock_data import commands, main_commands +from backend.utils.time import to_unix_time @pytest.fixture @@ -14,48 +13,16 @@ def db_engine(): @pytest.fixture -def unix_time(): - default_datetime = datetime.strptime("2021-01-01 00:00:00", "%Y-%m-%d %H:%M:%S") - return int(default_datetime.timestamp()) - - -@pytest.fixture -def main_commands(): - return [ - MainCommand( - id=1, - name="RTC Sync", - params="time", - format="int 7 bytes", - data_size=7, - total_size=8, - ), - MainCommand( - id=2, - name="Manually activate an emergency mode for a specified amount of time", - params="mode_state_number,time", - format="int 1 byte, int 7 bytes", - data_size=8, - total_size=9, - ), - ] - - -@pytest.fixture -def commands(unix_time): - return [ - Command(id=1, command_type=1, params=f"{unix_time}"), # RTC Sync for 2021-01-01 - Command( - id=2, command_type=2, params=f"1,{unix_time}" - ), # Emergency mode for 2021-01-01 - ] +def default_datetime(): + default_time = "2024-01-01T00:00:00" + return datetime.strptime(default_time, "%Y-%m-%dT%H:%M:%S") @pytest.fixture(scope="function", autouse=True) -def setup_db(db_engine, main_commands, commands): +def setup_db(db_engine, default_datetime): SQLModel.metadata.create_all(db_engine) with Session(db_engine) as session: - session.add_all(main_commands) + session.add_all(main_commands()) session.commit() - session.add_all(commands) + session.add_all(commands(to_unix_time(default_datetime))) session.commit() diff --git a/test/backend/test_setup.py b/test/backend/test_setup.py index 58b043a..9c58b73 100644 --- a/test/backend/test_setup.py +++ b/test/backend/test_setup.py @@ -3,6 +3,8 @@ from backend.data.data_models import MainCommand, Command from sqlmodel import Session +from backend.utils.time import to_unix_time + @pytest.mark.parametrize( "command_id, name, params, format, data_size, total_size", @@ -34,9 +36,9 @@ def test_main_command_setup( @pytest.mark.parametrize( "id, command_type, params", [(1, 1, "{unix_time}"), [2, 2, "1,{unix_time}"]] ) -def test_command_setup(db_engine, id, command_type, params, unix_time): +def test_command_setup(db_engine, id, command_type, params, default_datetime): with Session(db_engine) as session: command = session.get(Command, id) assert command assert command.command_type == command_type - assert command.params == params.format(unix_time=unix_time) + assert command.params == params.format(unix_time=to_unix_time(default_datetime)) From 863b3bffca93b60c948945ee8adcc4542932c85a Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 6 Oct 2024 12:44:56 -0400 Subject: [PATCH 20/27] Add logging and refactor setup --- backend/api/lifespan.py | 5 ++- backend/api/resources/command.py | 1 - backend/api/setup.py | 17 ++++++++ backend/main.py | 19 +-------- backend/utils/logging.py | 69 ++++++++++++++++++++++++++++++++ requirements.txt | 1 + setup.cfg | 1 + 7 files changed, 93 insertions(+), 20 deletions(-) create mode 100644 backend/api/setup.py create mode 100644 backend/utils/logging.py diff --git a/backend/api/lifespan.py b/backend/api/lifespan.py index 13c0ce6..2a52345 100644 --- a/backend/api/lifespan.py +++ b/backend/api/lifespan.py @@ -5,12 +5,14 @@ from backend.data.data_models import MainCommand from backend.data.engine import get_db from backend.data.mock_data import commands, main_commands +from backend.utils.logging import logger_setup from backend.utils.time import to_unix_time @asynccontextmanager async def lifespan(_: FastAPI): - print("Starting application") + """Lifecycle event for the FastAPI app.""" + logger_setup() SQLModel.metadata.create_all(get_db().connection()) default_time = "2024-01-01T00:00:00" default_datetime = datetime.strptime(default_time, "%Y-%m-%dT%H:%M:%S") @@ -19,7 +21,6 @@ async def lifespan(_: FastAPI): with get_db() as session: query = select(MainCommand).limit(1) # Check if the db is empty result = session.exec(query).first() - print(result) if result is None: session.add_all(main_commands()) session.commit() diff --git a/backend/api/resources/command.py b/backend/api/resources/command.py index b675ad3..b801d08 100644 --- a/backend/api/resources/command.py +++ b/backend/api/resources/command.py @@ -18,7 +18,6 @@ async def get_items(db: Session = Depends(get_db)): """ query = select(Command) items = db.exec(query).all() - print(f"Items: {items}") return {"data": items} diff --git a/backend/api/setup.py b/backend/api/setup.py new file mode 100644 index 0000000..29ac553 --- /dev/null +++ b/backend/api/setup.py @@ -0,0 +1,17 @@ +from backend.api.resources.command import command_router +from backend.api.resources.main_command import main_command_router +from backend.api.middlewares.cors_middleware import add_cors_middleware +from backend.api.middlewares.logger_middleware import LoggerMiddleware +from fastapi import FastAPI + + +def setup_routes(app: FastAPI) -> None: + """Adds the routes to the app""" + app.include_router(router=command_router, prefix="/commands") + app.include_router(router=main_command_router, prefix="/main-commands") + + +def setup_middlewares(app: FastAPI) -> None: + """Adds the middlewares to the app""" + add_cors_middleware(app) + app.add_middleware(LoggerMiddleware) diff --git a/backend/main.py b/backend/main.py index 6fcd9e8..2235c37 100644 --- a/backend/main.py +++ b/backend/main.py @@ -1,21 +1,6 @@ -from backend.api.resources.command import command_router -from backend.api.resources.main_command import main_command_router -from backend.api.lifespan import lifespan -from backend.api.middlewares.cors_middleware import add_cors_middleware -from backend.api.middlewares.logger_middleware import LoggerMiddleware from fastapi import FastAPI - - -def setup_routes(app: FastAPI) -> None: - """Adds the routes to the app""" - app.include_router(router=command_router, prefix="/commands") - app.include_router(router=main_command_router, prefix="/main-commands") - - -def setup_middlewares(app: FastAPI) -> None: - """Adds the middlewares to the app""" - add_cors_middleware(app) - app.add_middleware(LoggerMiddleware) +from backend.api.lifespan import lifespan +from backend.api.setup import setup_routes, setup_middlewares app = FastAPI(lifespan=lifespan) diff --git a/backend/utils/logging.py b/backend/utils/logging.py new file mode 100644 index 0000000..02fb57c --- /dev/null +++ b/backend/utils/logging.py @@ -0,0 +1,69 @@ +import sys +from typing import Final + +from loguru import logger + +DEFAULT_LOG_FORMAT: Final[ + str +] = """{time:YYYY-MM-DD at HH:mm:ss.SSSSZ} | \ +{level: <8} | \ +{name}:{function}:{line} \ +- {message} \ +""" + + +def logger_setup(*, enqueue: bool = False, diagnose: bool = True) -> None: + """ + Set up the logger and return it. + The logger will log everything to a file, info to stdout, and warnings and above to stderr. + @param enqueue - Whether to enqueue messages for asynchronous processing. + @param diagnose - Whether to enable diagnostic mode. + """ + # Remove any existing sinks + logger.remove() + + # Log everything to a file + logger_setup_file(enqueue=enqueue, diagnose=diagnose) + + # Log info to stdout + logger.add( + sys.stdout, + filter=lambda record: record["level"].name == "INFO", + format=DEFAULT_LOG_FORMAT, + level="INFO", + colorize=True, + enqueue=enqueue, + diagnose=diagnose, + ) + + # Log warnings and above to stderr + logger.add( + sys.stderr, + format=DEFAULT_LOG_FORMAT, + level="WARNING", + colorize=True, + enqueue=enqueue, + diagnose=diagnose, + ) + + +def logger_setup_file(*, enqueue: bool = False, diagnose: bool = True) -> None: + """Set up the logger to log everything to a file.""" + logger.add( + "gs_python.log", + serialize=True, + format=DEFAULT_LOG_FORMAT, + rotation="1 week", + retention="1 month", + enqueue=enqueue, + diagnose=diagnose, + ) + + +async def logger_close() -> None: + """ + Close the logger for the async applications. + If the logger was setup using `enqueue=True`, this function should be awaited before the application exits. + """ + await logger.complete() + logger.remove() # Clear existing sinks to prevent semaphore leakage diff --git a/requirements.txt b/requirements.txt index 21c0869..e924a03 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,6 +1,7 @@ # Regular dependencies fastapi[standard]==0.115.0 sqlmodel==0.0.22 +loguru==0.7.2 # Development dependencies httpx==0.27.2 diff --git a/setup.cfg b/setup.cfg index bfcfe66..7cd7253 100644 --- a/setup.cfg +++ b/setup.cfg @@ -10,6 +10,7 @@ packages = backend install_requires = fastapi>=0.115.0 sqlmodel>=0.0.22 + loguru>=0.7.2 python_requires = ~=3.10 zip_safe = no From b2142ee604ce464592716a54eb05afd9253a9595 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 6 Oct 2024 12:49:45 -0400 Subject: [PATCH 21/27] Finish setup of frontend --- frontend/src/display/table.tsx | 5 ++++- frontend/src/input/command_input.tsx | 6 +++--- 2 files changed, 7 insertions(+), 4 deletions(-) diff --git a/frontend/src/display/table.tsx b/frontend/src/display/table.tsx index 7a895bd..3d95a6a 100644 --- a/frontend/src/display/table.tsx +++ b/frontend/src/display/table.tsx @@ -16,7 +16,10 @@ const CommandTable = () => { }, []) const handleDelete = (id: number) => { - return () => { setCommands(commands.filter((value) => value.id != id)) } + return () => { + // TODO: (Member) Handle delete logic here + + } } return ( diff --git a/frontend/src/input/command_input.tsx b/frontend/src/input/command_input.tsx index 2ca901a..2cdbb5e 100644 --- a/frontend/src/input/command_input.tsx +++ b/frontend/src/input/command_input.tsx @@ -1,7 +1,7 @@ import "./command_input.css" const CommandInput = () => { - // Setup state and useEffect calls here + // TODO: (Member) Setup state and useEffect calls here const handleSubmit = () => { // TODO:(Member) Submit to your post endpoint @@ -13,13 +13,13 @@ const CommandInput = () => {
- {/* TODO: (Member) Display the list of commands based on the get commands request*/}
- {/*Add input handling here if the selected command has a param input*/} + {/* TODO: (Member) Add input handling here if the selected command has a param input*/}
From 9765b94d908fd6d7f3eaf39d95e3f7b349b6fa04 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 6 Oct 2024 13:19:40 -0400 Subject: [PATCH 22/27] Add comments and rename functions --- backend/api/resources/command.py | 9 +++++---- backend/data/data_models.py | 6 +++++- 2 files changed, 10 insertions(+), 5 deletions(-) diff --git a/backend/api/resources/command.py b/backend/api/resources/command.py index b801d08..c642785 100644 --- a/backend/api/resources/command.py +++ b/backend/api/resources/command.py @@ -10,7 +10,7 @@ @command_router.get("/", response_model=CommandListResponse) -async def get_items(db: Session = Depends(get_db)): +async def get_commands(db: Session = Depends(get_db)): """ Gets all the items @@ -22,7 +22,7 @@ async def get_items(db: Session = Depends(get_db)): @command_router.post("/", response_model=Command) -async def create_item(payload: CommandRequest): +async def create_command(payload: CommandRequest): """ Creates an item with the given payload and returns the payload with some other information @@ -32,11 +32,12 @@ async def create_item(payload: CommandRequest): # TODO: Implement this endpoint -@command_router.delete("/{id}") -async def delete_item(id: int): +@command_router.delete("/{id}", response_model=CommandListResponse) +async def delete_command(id: int): """ Deletes the item with the given id if it exists. Otherwise raises a 404 error. @param id: The id of the item to delete + @return returns the list of commands after deleting the item """ # TODO: Implement this endpoint diff --git a/backend/data/data_models.py b/backend/data/data_models.py index 777f6eb..6a9eab4 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -26,7 +26,11 @@ class MainCommand(SQLModel, table=True): @model_validator(mode="after") def validate_params_format(self): - """Check that params and format are both None or that the params and format have the same number of comma seperated values""" + """ + Check that params and format are both None or that the params and format have the same number of comma seperated values. + In either of these cases return self. Otherwise raise a ValueError. + """ + # TODO: (Member) Implement this method if self.params is None and self.format is None: return self if self.params is None: From 7ae3aabf39a87bb95a59e8ce74782b4a25337621 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 6 Oct 2024 13:45:31 -0400 Subject: [PATCH 23/27] Add model validation tests for MainCommand --- backend/data/base_model.py | 16 +++++++++++ backend/data/data_models.py | 7 +++-- test/backend/test_data.py | 57 +++++++++++++++++++++++++++++++++++++ 3 files changed, 77 insertions(+), 3 deletions(-) create mode 100644 backend/data/base_model.py diff --git a/backend/data/base_model.py b/backend/data/base_model.py new file mode 100644 index 0000000..2ebc20d --- /dev/null +++ b/backend/data/base_model.py @@ -0,0 +1,16 @@ +from sqlmodel import SQLModel +from sqlmodel._compat import get_config_value, set_config_value + + +class BaseSQLModel(SQLModel): + """ + Base SQL Model class. It performs validation on the model unlike the default SQLModel class with table=True. + """ + + def __init__(self, **data): + is_table = get_config_value(model=self, parameter="table", default=False) + set_config_value( + model=self, parameter="table", value=False + ) # Makes it validate the model + super().__init__(**data) + set_config_value(model=self, parameter="table", value=is_table) diff --git a/backend/data/data_models.py b/backend/data/data_models.py index 6a9eab4..f5dfe01 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -2,12 +2,13 @@ # NOTE: This file should not be modified from datetime import datetime from pydantic import model_validator -from sqlmodel import Field, SQLModel +from sqlmodel import Field +from backend.data.base_model import BaseSQLModel from backend.data.enums import CommandStatus -class MainCommand(SQLModel, table=True): +class MainCommand(BaseSQLModel, table=True): """ Main command model. This table represents all the possible commands that can be issued. @@ -44,7 +45,7 @@ def validate_params_format(self): return self -class Command(SQLModel, table=True): +class Command(BaseSQLModel, table=True): """ An instance of a MainCommand. This table holds the data related to actual commands sent from the ground station up to the OBC. diff --git a/test/backend/test_data.py b/test/backend/test_data.py index e69de29..2295bd1 100644 --- a/test/backend/test_data.py +++ b/test/backend/test_data.py @@ -0,0 +1,57 @@ +import pytest +from sqlmodel import Session +from backend.data.data_models import MainCommand + + +def test_main_command(): + main_command = MainCommand( + name="Test", + params="param1,param2", + format="int,int", + data_size=2, + total_size=2, + ) + assert main_command.name == "Test" + assert main_command.params == "param1,param2" + assert main_command.format == "int,int" + assert main_command.data_size == 2 + assert main_command.total_size == 2 + + +def test_main_command_no_params_and_no_format(): + main_command = MainCommand( + name="Test", + data_size=2, + total_size=2, + ) + assert main_command.name == "Test" + assert main_command.params is None + assert main_command.format is None + assert main_command.data_size == 2 + assert main_command.total_size == 2 + + +def test_main_command_no_format(db_engine): + with pytest.raises(ValueError), Session(db_engine) as session: + session.add( + MainCommand( + name="Test", + params="param1,param2", + data_size=2, + total_size=2, + ) + ) + session.commit() + + +def test_main_command_no_params(db_engine): + with pytest.raises(ValueError), Session(db_engine) as session: + session.add( + MainCommand( + name="Test", + format="int,int", + data_size=2, + total_size=2, + ) + ) + session.commit() From c67647e8592ec318a1fb76a3e9f7fdbe5c2f720f Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 3 Nov 2024 20:38:33 -0500 Subject: [PATCH 24/27] Add get commands test --- backend/api/lifespan.py | 30 ++++++----- backend/api/resources/command.py | 6 +-- backend/api/resources/main_command.py | 2 +- backend/data/mock_data.py | 4 +- backend/domain/__init__.py | 0 backend/domain/base_service.py | 4 -- backend/domain/command_service.py | 0 backend/domain/iservice.py | 9 ---- requirements.txt | 3 +- test/backend/conftest.py | 75 ++++++++++++++++++++++++--- test/backend/test_api.py | 6 +++ test/backend/test_data.py | 17 +++--- test/backend/test_setup.py | 35 +++++++------ 13 files changed, 124 insertions(+), 67 deletions(-) delete mode 100644 backend/domain/__init__.py delete mode 100644 backend/domain/base_service.py delete mode 100644 backend/domain/command_service.py delete mode 100644 backend/domain/iservice.py diff --git a/backend/api/lifespan.py b/backend/api/lifespan.py index 2a52345..c3a46aa 100644 --- a/backend/api/lifespan.py +++ b/backend/api/lifespan.py @@ -1,7 +1,7 @@ from contextlib import asynccontextmanager from datetime import datetime from fastapi import FastAPI -from sqlmodel import SQLModel, select +from sqlmodel import SQLModel, Session, select from backend.data.data_models import MainCommand from backend.data.engine import get_db from backend.data.mock_data import commands, main_commands @@ -9,21 +9,25 @@ from backend.utils.time import to_unix_time -@asynccontextmanager -async def lifespan(_: FastAPI): - """Lifecycle event for the FastAPI app.""" +def create_startup(session: Session) -> None: logger_setup() - SQLModel.metadata.create_all(get_db().connection()) + SQLModel.metadata.create_all(session.connection()) default_time = "2024-01-01T00:00:00" default_datetime = datetime.strptime(default_time, "%Y-%m-%dT%H:%M:%S") unix_time = to_unix_time(default_datetime) # Setup the db with mock data - with get_db() as session: - query = select(MainCommand).limit(1) # Check if the db is empty - result = session.exec(query).first() - if result is None: - session.add_all(main_commands()) - session.commit() - session.add_all(commands(unix_time)) - session.commit() + query = select(MainCommand).limit(1) # Check if the db is empty + result = session.exec(query).first() + if result is None: + session.add_all(main_commands()) + session.commit() + session.add_all(commands(unix_time)) + session.commit() + + +@asynccontextmanager +async def lifespan(_: FastAPI): + """Lifecycle event for the FastAPI app.""" + create_startup(get_db()) yield + print("Closing lifespan") diff --git a/backend/api/resources/command.py b/backend/api/resources/command.py index c642785..bed5109 100644 --- a/backend/api/resources/command.py +++ b/backend/api/resources/command.py @@ -10,7 +10,7 @@ @command_router.get("/", response_model=CommandListResponse) -async def get_commands(db: Session = Depends(get_db)): +def get_commands(db: Session = Depends(get_db)): """ Gets all the items @@ -22,7 +22,7 @@ async def get_commands(db: Session = Depends(get_db)): @command_router.post("/", response_model=Command) -async def create_command(payload: CommandRequest): +def create_command(payload: CommandRequest): """ Creates an item with the given payload and returns the payload with some other information @@ -33,7 +33,7 @@ async def create_command(payload: CommandRequest): @command_router.delete("/{id}", response_model=CommandListResponse) -async def delete_command(id: int): +def delete_command(id: int): """ Deletes the item with the given id if it exists. Otherwise raises a 404 error. diff --git a/backend/api/resources/main_command.py b/backend/api/resources/main_command.py index ae03be2..fecd3d7 100644 --- a/backend/api/resources/main_command.py +++ b/backend/api/resources/main_command.py @@ -9,7 +9,7 @@ @main_command_router.get("/", response_model=MainCommandListResponse) -async def get_main_commands(db: Session = Depends(get_db)): +def get_main_commands(db: Session = Depends(get_db)): """ Gets all the main commands that can be created. diff --git a/backend/data/mock_data.py b/backend/data/mock_data.py index dfeaa5a..110d399 100644 --- a/backend/data/mock_data.py +++ b/backend/data/mock_data.py @@ -3,10 +3,10 @@ def commands(unix_time: int) -> list[Command]: return [ - Command(command_type=1, params=f"{unix_time}"), # id=1, RTC Sync for 2021-01-01 + Command(command_type=1, params=f"{unix_time}"), # id=1, RTC Sync for unix time Command( command_type=2, params=f"1,{unix_time}" - ), # id=2, Emergency mode for 2021-01-01 + ), # id=2, Emergency mode for unix time ] diff --git a/backend/domain/__init__.py b/backend/domain/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/domain/base_service.py b/backend/domain/base_service.py deleted file mode 100644 index cd07543..0000000 --- a/backend/domain/base_service.py +++ /dev/null @@ -1,4 +0,0 @@ -from backend.domain.iservice import IService - - -class BaseService(IService): ... diff --git a/backend/domain/command_service.py b/backend/domain/command_service.py deleted file mode 100644 index e69de29..0000000 diff --git a/backend/domain/iservice.py b/backend/domain/iservice.py deleted file mode 100644 index a9d0026..0000000 --- a/backend/domain/iservice.py +++ /dev/null @@ -1,9 +0,0 @@ -from abc import ABC, abstractmethod -from typing import Generic, TypeVar - -T = TypeVar("T") - - -class IService(ABC, Generic[T]): - @abstractmethod - def get(self, id: int) -> T: ... diff --git a/requirements.txt b/requirements.txt index e924a03..9a3876a 100644 --- a/requirements.txt +++ b/requirements.txt @@ -4,6 +4,5 @@ sqlmodel==0.0.22 loguru==0.7.2 # Development dependencies -httpx==0.27.2 -pytest==7.4.0 +pytest==8.3.3 pytest-cov==4.1.0 diff --git a/test/backend/conftest.py b/test/backend/conftest.py index 420b071..75015d6 100644 --- a/test/backend/conftest.py +++ b/test/backend/conftest.py @@ -1,15 +1,29 @@ +from contextlib import asynccontextmanager from datetime import datetime -from sqlmodel import SQLModel, Session, create_engine +from typing import Final +from fastapi import FastAPI +from fastapi.testclient import TestClient +from sqlmodel import SQLModel, Session, create_engine, delete import pytest +from backend.api.lifespan import create_startup +from backend.api.setup import setup_middlewares, setup_routes +from backend.data.data_models import Command, MainCommand +from backend.data.engine import get_db from backend.data.mock_data import commands, main_commands from backend.utils.time import to_unix_time +from json import loads +from sqlmodel.pool import StaticPool +TESTING_SQL_PATH: Final[str] = "sqlite:///testing.db" + +def get_mock_db() -> Session: + engine = create_engine(TESTING_SQL_PATH, connect_args={"check_same_thread": False}, poolclass=StaticPool) + with Session(engine) as session: + return session @pytest.fixture -def db_engine(): - sqlite_file_name = "sqlite://" # In memory db for testing - engine = create_engine(sqlite_file_name) - return engine +def mock_db() -> Session: + return get_mock_db() @pytest.fixture @@ -18,11 +32,56 @@ def default_datetime(): return datetime.strptime(default_time, "%Y-%m-%dT%H:%M:%S") +@asynccontextmanager +async def lifespan(_: FastAPI): + db = get_mock_db() + create_startup(db) + yield + del_main_command = delete(MainCommand) + db.exec(del_main_command) + db.commit() + del_command = delete(Command) + db.exec(del_command) + db.commit() + + +@pytest.fixture +def fastapi_app(): + app = FastAPI(lifespan=lifespan) + app.dependency_overrides[get_db] = get_mock_db + setup_routes(app) + setup_middlewares(app) + return app + +@pytest.fixture +def fastapi_test_client(fastapi_app): + return TestClient(fastapi_app) + +@pytest.fixture +def commands_json(default_datetime): + commands_with_id = [] + for i, val in enumerate(commands(to_unix_time(default_datetime)), start=1): + val.id = i + serialize_command = val.model_dump_json() + command = loads(serialize_command) + commands_with_id.append(command) + return commands_with_id + + @pytest.fixture(scope="function", autouse=True) -def setup_db(db_engine, default_datetime): - SQLModel.metadata.create_all(db_engine) - with Session(db_engine) as session: +def setup_db(mock_db: Session, default_datetime): + SQLModel.metadata.create_all(mock_db.connection()) + with mock_db as session: session.add_all(main_commands()) session.commit() session.add_all(commands(to_unix_time(default_datetime))) session.commit() + yield + del_main_command = delete(MainCommand) + session.exec(del_main_command) + session.commit() + del_command = delete(Command) + session.exec(del_command) + session.commit() + + diff --git a/test/backend/test_api.py b/test/backend/test_api.py index e69de29..3b60115 100644 --- a/test/backend/test_api.py +++ b/test/backend/test_api.py @@ -0,0 +1,6 @@ + +def test_get_commands(fastapi_test_client, commands_json): + with fastapi_test_client as client: + response = client.get("/commands/") + assert response.status_code == 200 + assert response.json() == {"data": commands_json} diff --git a/test/backend/test_data.py b/test/backend/test_data.py index 2295bd1..a40d0ba 100644 --- a/test/backend/test_data.py +++ b/test/backend/test_data.py @@ -1,5 +1,4 @@ import pytest -from sqlmodel import Session from backend.data.data_models import MainCommand @@ -31,9 +30,9 @@ def test_main_command_no_params_and_no_format(): assert main_command.total_size == 2 -def test_main_command_no_format(db_engine): - with pytest.raises(ValueError), Session(db_engine) as session: - session.add( +def test_main_command_no_format(mock_db): + with pytest.raises(ValueError): + mock_db.add( MainCommand( name="Test", params="param1,param2", @@ -41,12 +40,12 @@ def test_main_command_no_format(db_engine): total_size=2, ) ) - session.commit() + mock_db.commit() -def test_main_command_no_params(db_engine): - with pytest.raises(ValueError), Session(db_engine) as session: - session.add( +def test_main_command_no_params(mock_db): + with pytest.raises(ValueError): + mock_db.add( MainCommand( name="Test", format="int,int", @@ -54,4 +53,4 @@ def test_main_command_no_params(db_engine): total_size=2, ) ) - session.commit() + mock_db.commit() diff --git a/test/backend/test_setup.py b/test/backend/test_setup.py index 9c58b73..d5f4aae 100644 --- a/test/backend/test_setup.py +++ b/test/backend/test_setup.py @@ -1,7 +1,7 @@ # Test to make sure the conftest.py setup is correct import pytest +from sqlmodel import Session, select from backend.data.data_models import MainCommand, Command -from sqlmodel import Session from backend.utils.time import to_unix_time @@ -21,24 +21,27 @@ ], ) def test_main_command_setup( - db_engine, command_id, name, params, format, data_size, total_size + mock_db: Session, command_id, name, params, format, data_size, total_size ): - with Session(db_engine) as session: - main_command = session.get(MainCommand, command_id) - assert main_command - assert main_command.name == name - assert main_command.params == params - assert main_command.format == format - assert main_command.data_size == data_size - assert main_command.total_size == total_size + main_command = mock_db.get(MainCommand, command_id) + assert main_command + assert main_command.name == name + assert main_command.params == params + assert main_command.format == format + assert main_command.data_size == data_size + assert main_command.total_size == total_size + query = select(MainCommand) + all_main_commands = mock_db.exec(query).all() + assert len(all_main_commands) == 2 @pytest.mark.parametrize( "id, command_type, params", [(1, 1, "{unix_time}"), [2, 2, "1,{unix_time}"]] ) -def test_command_setup(db_engine, id, command_type, params, default_datetime): - with Session(db_engine) as session: - command = session.get(Command, id) - assert command - assert command.command_type == command_type - assert command.params == params.format(unix_time=to_unix_time(default_datetime)) +def test_command_setup(mock_db, id, command_type, params, default_datetime): + command = mock_db.get(Command, id) + assert command + assert command.command_type == command_type + assert command.params == params.format(unix_time=to_unix_time(default_datetime)) + + From b12446080487c5dcf2b37edbb275a824d5d4e0e2 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 3 Nov 2024 21:16:07 -0500 Subject: [PATCH 25/27] Add create test --- backend/api/models/response_model.py | 7 ++++++ backend/api/resources/command.py | 17 ++++++++++---- test/backend/test_api.py | 34 ++++++++++++++++++++++++---- 3 files changed, 49 insertions(+), 9 deletions(-) diff --git a/backend/api/models/response_model.py b/backend/api/models/response_model.py index d541eb8..106b18b 100644 --- a/backend/api/models/response_model.py +++ b/backend/api/models/response_model.py @@ -17,3 +17,10 @@ class MainCommandListResponse(BaseModel): """ data: list[MainCommand] + + +class CommandSingleResponse(BaseModel): + """ + Single command + """ + data: Command diff --git a/backend/api/resources/command.py b/backend/api/resources/command.py index bed5109..9ac8af3 100644 --- a/backend/api/resources/command.py +++ b/backend/api/resources/command.py @@ -1,11 +1,12 @@ -from fastapi import APIRouter, Depends +from fastapi import APIRouter, Depends, params from sqlmodel import Session, select from backend.api.models.request_model import CommandRequest -from backend.api.models.response_model import CommandListResponse +from backend.api.models.response_model import CommandListResponse, CommandSingleResponse from backend.data.data_models import Command from backend.data.engine import get_db +# Prefix: "/commands" command_router = APIRouter(tags=["Commands"]) @@ -21,15 +22,21 @@ def get_commands(db: Session = Depends(get_db)): return {"data": items} -@command_router.post("/", response_model=Command) -def create_command(payload: CommandRequest): +@command_router.post("/", response_model=CommandSingleResponse) +def create_command(payload: CommandRequest, db: Session = Depends(get_db)): """ Creates an item with the given payload and returns the payload with some other information @param payload: The data used to create an item - @return returns the data with some other information + @return returns a json object with field of "data" under which there is the payload with some other information """ # TODO: Implement this endpoint + command = Command(command_type=payload.command_type, params=payload.params) + db.add(command) + db.commit() + db.refresh(command) + return {"data":command} + @command_router.delete("/{id}", response_model=CommandListResponse) diff --git a/test/backend/test_api.py b/test/backend/test_api.py index 3b60115..21e622f 100644 --- a/test/backend/test_api.py +++ b/test/backend/test_api.py @@ -1,6 +1,32 @@ -def test_get_commands(fastapi_test_client, commands_json): +from fastapi.testclient import TestClient + +from backend.api.models.request_model import CommandRequest +from backend.data.enums import CommandStatus + + +def test_get_commands(fastapi_test_client: TestClient, commands_json): with fastapi_test_client as client: - response = client.get("/commands/") - assert response.status_code == 200 - assert response.json() == {"data": commands_json} + res = client.get("/commands/") + assert res.status_code == 200 + assert res.json() == {"data": commands_json} + + +def test_create_command(fastapi_test_client: TestClient): + command = CommandRequest(command_type=1, params="123456789") + model_dump = command.model_dump() + print(model_dump) + with fastapi_test_client as client: + res = client.post("/commands/", json=model_dump, headers={"Content-Type": "application/json"}) + # res = client.post("/commands/", json=command) + res.raise_for_status() + assert res.status_code == 200 + result = res.json().get("data") + assert result is not None + assert result.get("id") == 3 + assert result.get("command_type") == 1 + assert result.get("status") == CommandStatus.PENDING.value + assert result.get("params") == "123456789" + # TODO: Figure out a better way to check the times + assert result.get("created_on") + assert result.get("updated_on") From 1af7f778cefe3d1e96a35ae3cd0b110ce7244b97 Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 3 Nov 2024 21:42:56 -0500 Subject: [PATCH 26/27] Add delete and main command tests --- backend/api/resources/command.py | 16 +++++++-- backend/api/resources/main_command.py | 1 + test/backend/test_api.py | 51 ++++++++++++++++++++++++--- 3 files changed, 61 insertions(+), 7 deletions(-) diff --git a/backend/api/resources/command.py b/backend/api/resources/command.py index 9ac8af3..14bfbcc 100644 --- a/backend/api/resources/command.py +++ b/backend/api/resources/command.py @@ -1,4 +1,5 @@ -from fastapi import APIRouter, Depends, params +from fastapi import APIRouter, Depends +from fastapi.exceptions import HTTPException from sqlmodel import Session, select from backend.api.models.request_model import CommandRequest @@ -40,7 +41,7 @@ def create_command(payload: CommandRequest, db: Session = Depends(get_db)): @command_router.delete("/{id}", response_model=CommandListResponse) -def delete_command(id: int): +def delete_command(id: int, db: Session = Depends(get_db)): """ Deletes the item with the given id if it exists. Otherwise raises a 404 error. @@ -48,3 +49,14 @@ def delete_command(id: int): @return returns the list of commands after deleting the item """ # TODO: Implement this endpoint + statement = select(Command).where(Command.id == id) + results = db.exec(statement).all() + if len(results) == 0: + raise HTTPException(status_code=404) + elif len(results) != 1: + raise HTTPException(status_code=400) + db.delete(results[0]) + db.commit() + statement = select(Command) + results = db.exec(statement).all() + return {"data": results} diff --git a/backend/api/resources/main_command.py b/backend/api/resources/main_command.py index fecd3d7..7dd8e2e 100644 --- a/backend/api/resources/main_command.py +++ b/backend/api/resources/main_command.py @@ -5,6 +5,7 @@ from backend.data.data_models import MainCommand from backend.data.engine import get_db +# Prefix: /main-commands main_command_router = APIRouter(tags=["Main Commands"]) diff --git a/test/backend/test_api.py b/test/backend/test_api.py index 21e622f..c859834 100644 --- a/test/backend/test_api.py +++ b/test/backend/test_api.py @@ -1,8 +1,7 @@ - from fastapi.testclient import TestClient - from backend.api.models.request_model import CommandRequest from backend.data.enums import CommandStatus +from backend.utils.time import to_unix_time def test_get_commands(fastapi_test_client: TestClient, commands_json): @@ -15,11 +14,8 @@ def test_get_commands(fastapi_test_client: TestClient, commands_json): def test_create_command(fastapi_test_client: TestClient): command = CommandRequest(command_type=1, params="123456789") model_dump = command.model_dump() - print(model_dump) with fastapi_test_client as client: res = client.post("/commands/", json=model_dump, headers={"Content-Type": "application/json"}) - # res = client.post("/commands/", json=command) - res.raise_for_status() assert res.status_code == 200 result = res.json().get("data") assert result is not None @@ -30,3 +26,48 @@ def test_create_command(fastapi_test_client: TestClient): # TODO: Figure out a better way to check the times assert result.get("created_on") assert result.get("updated_on") + +def test_delete_command_fail(fastapi_test_client: TestClient): + with fastapi_test_client as client: + res = client.delete("/commands/0") # Should never have an id of 0 in the db + assert res.status_code == 404 + +def test_delete_command(fastapi_test_client: TestClient, default_datetime): + with fastapi_test_client as client: + res = client.delete("/commands/1") + assert res.status_code == 200 + result = res.json().get("data") + assert result is not None + assert len(result) == 1 # Should only have 1 element in db + result = result[0] + assert result.get("id") == 2 + assert result.get("command_type") == 2 + assert result.get("status") == CommandStatus.PENDING.value + assert result.get("params") == f"1,{to_unix_time(default_datetime)}" + # TODO: Figure out a better way to check the times + assert result.get("created_on") + assert result.get("updated_on") + + +def test_main_commands(fastapi_test_client: TestClient): + with fastapi_test_client as client: + res = client.get("/main-commands/") + assert res.status_code == 200 + result = res.json().get("data") + assert result is not None + assert len(result) == 2 + main_command = result[0] + assert main_command.get("id") == 1 + assert main_command.get("name") == "RTC Sync" + assert main_command.get("params") == "time" + assert main_command.get("format") == "int 7 bytes" + assert main_command.get("data_size") == 7 + assert main_command.get("total_size") == 8 + main_command = result[1] + assert main_command.get("id") == 2 + assert main_command.get("name") == "Manually activate an emergency mode for a specified amount of time" + assert main_command.get("params") == "mode_state_number,time" + assert main_command.get("format") == "int 1 byte, int 7 bytes" + assert main_command.get("data_size") == 8 + assert main_command.get("total_size") == 9 + From 36634bbc70a779e5b8bda9db17c6904ad8b657ea Mon Sep 17 00:00:00 2001 From: Yarik Popov <89220488+Yarik-Popov@users.noreply.github.com> Date: Sun, 3 Nov 2024 21:46:35 -0500 Subject: [PATCH 27/27] Clean up --- backend/api/resources/command.py | 24 ++++-------------------- backend/data/data_models.py | 10 ---------- pyproject.toml | 2 +- requirements.txt | 1 - 4 files changed, 5 insertions(+), 32 deletions(-) diff --git a/backend/api/resources/command.py b/backend/api/resources/command.py index 14bfbcc..7e3d5a7 100644 --- a/backend/api/resources/command.py +++ b/backend/api/resources/command.py @@ -24,39 +24,23 @@ def get_commands(db: Session = Depends(get_db)): @command_router.post("/", response_model=CommandSingleResponse) -def create_command(payload: CommandRequest, db: Session = Depends(get_db)): +def create_command(payload: CommandRequest): """ Creates an item with the given payload and returns the payload with some other information @param payload: The data used to create an item @return returns a json object with field of "data" under which there is the payload with some other information """ - # TODO: Implement this endpoint - command = Command(command_type=payload.command_type, params=payload.params) - db.add(command) - db.commit() - db.refresh(command) - return {"data":command} + # TODO:(Member) Implement this endpoint @command_router.delete("/{id}", response_model=CommandListResponse) -def delete_command(id: int, db: Session = Depends(get_db)): +def delete_command(id: int): """ Deletes the item with the given id if it exists. Otherwise raises a 404 error. @param id: The id of the item to delete @return returns the list of commands after deleting the item """ - # TODO: Implement this endpoint - statement = select(Command).where(Command.id == id) - results = db.exec(statement).all() - if len(results) == 0: - raise HTTPException(status_code=404) - elif len(results) != 1: - raise HTTPException(status_code=400) - db.delete(results[0]) - db.commit() - statement = select(Command) - results = db.exec(statement).all() - return {"data": results} + # TODO:(Member) Implement this endpoint diff --git a/backend/data/data_models.py b/backend/data/data_models.py index f5dfe01..33a9512 100644 --- a/backend/data/data_models.py +++ b/backend/data/data_models.py @@ -32,16 +32,6 @@ def validate_params_format(self): In either of these cases return self. Otherwise raise a ValueError. """ # TODO: (Member) Implement this method - if self.params is None and self.format is None: - return self - if self.params is None: - raise ValueError("params is None but format is not None") - if self.format is None: - raise ValueError("format is None but params is not None") - if self.params.count(",") != self.format.count(","): - raise ValueError( - "params and format must have the same number of comma seperated values" - ) return self diff --git a/pyproject.toml b/pyproject.toml index 07a622e..3f3aebb 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -4,5 +4,5 @@ requires = ["setuptools>=59.0", "wheel"] build-backend = "setuptools.build_meta" [tool.pytest.ini_options] -addopts = "--cov=backend -v" +addopts = "-vvv" testpaths = ["test/backend"] diff --git a/requirements.txt b/requirements.txt index 9a3876a..c75b2bb 100644 --- a/requirements.txt +++ b/requirements.txt @@ -5,4 +5,3 @@ loguru==0.7.2 # Development dependencies pytest==8.3.3 -pytest-cov==4.1.0