From fd6d36a99ca850a4164ccaa3e61b06a0f1defee1 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Sat, 24 May 2025 15:03:57 +0300 Subject: [PATCH 01/52] Rename README to Assignment --- README.md => Assignment.md | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename README.md => Assignment.md (100%) diff --git a/README.md b/Assignment.md similarity index 100% rename from README.md rename to Assignment.md From 99e3ee5ccf81f481f64872284e8b272ca1762b37 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Sat, 24 May 2025 15:06:47 +0300 Subject: [PATCH 02/52] Initial skeleton commit --- .gitignore | 2 + README.md | 93 +++++++++++++++ build/Dockerfile | 38 +++++++ build/Dockerfile.dev | 26 +++++ build/Dockerfile.utilities | 19 ++++ cmd/http/main.go | 65 +++++++++++ deployment/docker-compose.yml | 42 +++++++ examples/healthCheck.http | 4 + examples/token.http | 9 ++ go.mod | 27 +++++ go.sum | 97 ++++++++++++++++ internal/core/domain/jwtClaims.go | 16 +++ internal/core/domain/user.go | 23 ++++ internal/core/services/jwtService.go | 30 +++++ internal/handlers/healthCheckHandler.go | 57 ++++++++++ internal/handlers/jwtClaimsHandler.go | 145 ++++++++++++++++++++++++ internal/handlers/middleware.go | 37 ++++++ internal/repositories/userDao.go | 9 ++ pkg/auth/authMech.go | 62 ++++++++++ pkg/errors/errors.go | 61 ++++++++++ pkg/logger/loggerInterface.go | 9 ++ pkg/logger/logrus.go | 39 +++++++ pkg/server/routes.go | 33 ++++++ pkg/server/server.go | 65 +++++++++++ script/Makefile | 89 +++++++++++++++ 25 files changed, 1097 insertions(+) create mode 100644 .gitignore create mode 100644 README.md create mode 100755 build/Dockerfile create mode 100755 build/Dockerfile.dev create mode 100755 build/Dockerfile.utilities create mode 100644 cmd/http/main.go create mode 100755 deployment/docker-compose.yml create mode 100644 examples/healthCheck.http create mode 100644 examples/token.http create mode 100644 go.mod create mode 100644 go.sum create mode 100644 internal/core/domain/jwtClaims.go create mode 100644 internal/core/domain/user.go create mode 100644 internal/core/services/jwtService.go create mode 100644 internal/handlers/healthCheckHandler.go create mode 100644 internal/handlers/jwtClaimsHandler.go create mode 100644 internal/handlers/middleware.go create mode 100644 internal/repositories/userDao.go create mode 100644 pkg/auth/authMech.go create mode 100644 pkg/errors/errors.go create mode 100644 pkg/logger/loggerInterface.go create mode 100644 pkg/logger/logrus.go create mode 100644 pkg/server/routes.go create mode 100644 pkg/server/server.go create mode 100644 script/Makefile diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..451fb35 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +.env +.idea \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..93a0ae8 --- /dev/null +++ b/README.md @@ -0,0 +1,93 @@ +# Platform Go Challenge + +--- + +## Description + +Service that provides a REST API offering CRUD operations for adding, removing, editing +and getting a User's favourite Assets (Chart, Insight, Audience) + +--- + +## Run + +`cd script && make start-app` + +* This command will start the app with `localhost` address and `:8080` port (specified in build/Dockerfile.dev and .env) + +`cd script && make migrate-data-small` + +* This command will add to DB 10 Users and 3 Assets per User for testing reasons + +`cd script && make migrate-data-large` + +* This command will add to DB 10 Users and 10.000 Assets per User for testing reasons + +Then you can add, edit, remove and get User's favourites like the examples in /examples +directory. To generate the needed Bearer token, please call /token endpoint with username +& password like in the example + +--- + +## Makefile Commands + +| Command | Usage | +|---------------------------------|------------------------------------------------------------------------| +| start-app | `Start app` | +| kill-app | `Stop app` | +| rebuild-app | `Rebuild app` | +| tests-all | `Run both unit and integration tests` | +| tests-benchmark | `Run benchmark tests` | +| tests-unit | `Run unit tests ` | +| tests-file FILE={filePath} | `Run specific file test` | +| generate-mock FILE={filePath} | `Generate mock for a specific file` | +| run-linter | `Runs linter` | +| generate-swagger-files | `Generates swagger.json definitions in Docs dir` | +| tests-package PACKAGE={package} | `Run specific package test` | +| tests-all-with-coverage | `Run both unit and integration tests via docker with coverage details` | +| migrate-data-small | `Run migration of a small dataset Users & Assets in the DB` | +| migrate-data-large | `Run migration of a larger dataset Users & Assets in the DB` | + +* All these are executed through docker containers +* In order to execute makefile commands type **make** plus a command from the table above + + make {command} + +--- + +## Notes + +1. .env is pushed to Git only for the Assessment purpose. Config and .env files should never be tracked. +2. There are three Dockerfile files. + 1. Dockerfile is the normal, production one + 2. Dockerfile.dev is for setting up a remote debugger Delve + 3. Dockerfile.utilities is for building a docker for "utilities" like running tests, + linting etc +3. Asset types' fields and their validation was done very simply for the scope of this assessment +4. As described at the assignment, the User sees a list of Assets and adds some to their + favorites (by starring them for example). So the Assets are existing. This is why + the AddAsset() takes just IDs and just associates a User with an Asset, and not create it. + +## Known Issues + +1. Unit tests for adding, removing, editing assets are done only for Charts and only for happy paths. +2. JWT mechanism just requires a fake username and password to generate a JWT token and does NOT do + actual login due to lack of time. Also no test created for it + +## Performance + +If Users favourites data were causing performance issues, I would investigate the following solutions: + +1. Add a cache to the repository layer, over the DB and keep there the User's favourite Assets + in order to have faster access to it +1. Limit Payload by implementing Pagination +1. Limit Payload by compressing the data with Accept-Encoding: gzip or compress +1. Add indexes in DB. For example we could add indexed to the Users-Assets tables e.g "users_charts" on user_id + and chart_id + +* All these solutions would need to be discussed with the Engineering team to find the most suitable as + they all have their pros and cons + +## Security + +1. JWT mechanism added for Authentication and Authorization (incomplete - see Known Issues) \ No newline at end of file diff --git a/build/Dockerfile b/build/Dockerfile new file mode 100755 index 0000000..957c1b6 --- /dev/null +++ b/build/Dockerfile @@ -0,0 +1,38 @@ +# Start from golang base image +FROM golang:1.19.3-alpine3.16 as builder + +# Install git. +# Git is required for fetching the dependencies. +RUN apk update && apk add --no-cache git build-base bash + +# Set the current working directory inside the container +WORKDIR /app + +# Copy go mod and sum files +COPY ../go.mod go.sum ./ + +# Download all dependencies. Dependencies will be cached if the go.mod and the go.sum files are not changed +RUN go mod download + +# Copy the source from the current directory to the working Directory inside the container +COPY .. . + +# Build the Go app +RUN GO111MODULE=on CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o main ./cmd/http/main.go + +# Start a new stage from scratch +FROM alpine:latest +RUN apk --no-cache add ca-certificates + +WORKDIR /app/ + +# Copy the Pre-built binary file from the previous stage. Observe we also copied the .env file +COPY --from=builder /app/main . +COPY --from=builder /app/.env . + +# Expose port 8080 to the outside world +EXPOSE 8080 + +#Command to run the executable +RUN chmod +x ./main +CMD ["./main"] \ No newline at end of file diff --git a/build/Dockerfile.dev b/build/Dockerfile.dev new file mode 100755 index 0000000..a44de73 --- /dev/null +++ b/build/Dockerfile.dev @@ -0,0 +1,26 @@ +FROM golang:1.19.3-alpine3.16 as builder + +WORKDIR /app + +COPY ../go.mod go.sum ./ +RUN go mod download +RUN go install github.com/joho/godotenv/cmd/godotenv@v1.4.0 +RUN go install github.com/go-delve/delve/cmd/dlv@latest + +FROM golang:1.19.3-alpine3.16 + +RUN apk update +RUN apk add build-base bash + +COPY --from=builder /go /go + +WORKDIR /app + +COPY .. . + +RUN GOOS=linux go build -gcflags='all=-N -l' -tags musl -a -installsuffix cgo -o main ./cmd/http/main.go + +EXPOSE 8080 +EXPOSE 40000 + +CMD ["dlv", "--listen=:40000", "--headless=true", "--api-version=2", "--accept-multiclient", "--continue=true", "exec", "main"] \ No newline at end of file diff --git a/build/Dockerfile.utilities b/build/Dockerfile.utilities new file mode 100755 index 0000000..ae6f5b4 --- /dev/null +++ b/build/Dockerfile.utilities @@ -0,0 +1,19 @@ +FROM golang:1.19.3-alpine3.16 + +# Install git + +RUN apk update && apk add --no-cache git build-base bash + +WORKDIR /app + +COPY ../go.mod go.sum ./ + +RUN go mod download + +# Copy the source from the current directory to the working Directory inside the container +COPY .. . + +RUN go install github.com/golang/mock/mockgen@v1.6.0 +RUN go install github.com/joho/godotenv/cmd/godotenv@v1.4.0 +RUN wget -O- -nv https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b $(go env GOPATH)/bin v1.50.1 +RUN go install github.com/go-swagger/go-swagger/cmd/swagger@v0.30.4 \ No newline at end of file diff --git a/cmd/http/main.go b/cmd/http/main.go new file mode 100644 index 0000000..6357b35 --- /dev/null +++ b/cmd/http/main.go @@ -0,0 +1,65 @@ +package main + +import ( + "context" + "fmt" + "github.com/gorilla/mux" + "github.com/joho/godotenv" + "github.com/loukaspe/platform-go-challenge/internal/repositories" + "github.com/loukaspe/platform-go-challenge/pkg/logger" + "github.com/loukaspe/platform-go-challenge/pkg/server" + log "github.com/sirupsen/logrus" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "net/http" + "os" +) + +func main() { + getEnv() + + logger := logger.NewLogger(context.Background()) + router := mux.NewRouter() + httpServer := &http.Server{ + Addr: os.Getenv("SERVER_ADDR"), + Handler: router, + } + db := getDB() + + server := server.NewServer(db, router, httpServer, logger) + + server.Run() +} + +func getDB() *gorm.DB { + dbDsn := fmt.Sprintf( + "host=%s port=%s user=%s dbname=%s sslmode=disable password=%s TimeZone=Europe/Athens", + os.Getenv("DB_HOST"), + os.Getenv("DB_PORT"), + os.Getenv("DB_USER"), + os.Getenv("DB_NAME"), + os.Getenv("DB_PASSWORD"), + ) + db, err := gorm.Open(postgres.Open(dbDsn), &gorm.Config{}) + if err != nil { + log.Fatal("Cannot connect to database: ", err) + } + + // Drops added in order to start with clean DB on App start for + // assessment reasons + db.Migrator().DropTable("users") + + err = db.AutoMigrate(&repositories.User{}) + if err != nil { + log.Fatal("cannot migrate user table") + } + + return db +} + +func getEnv() { + err := godotenv.Load("./config/.env") + if err != nil { + log.Fatalf("Error getting env, not comming through %v", err) + } +} diff --git a/deployment/docker-compose.yml b/deployment/docker-compose.yml new file mode 100755 index 0000000..13a7d04 --- /dev/null +++ b/deployment/docker-compose.yml @@ -0,0 +1,42 @@ +version: '3' + +services: + postgres-db: + image: postgres:latest +# container_name: postgresdb + environment: + - POSTGRES_USER=${DB_USER} + - POSTGRES_PASSWORD=${DB_PASSWORD} + - POSTGRES_DB=${DB_NAME} + - DATABASE_HOST=${DB_HOST} + ports: + - '5432:5432' + volumes: + - database_postgres:/var/lib/postgresql/data + pgadmin: + image: dpage/pgadmin4 +# container_name: pgadmin_container + environment: + PGADMIN_DEFAULT_EMAIL: ${PGADMIN_DEFAULT_EMAIL} + PGADMIN_DEFAULT_PASSWORD: ${PGADMIN_DEFAULT_PASSWORD} + depends_on: + - postgres-db + ports: + - "5050:80" + app: +# container_name: platform-go-challenge-app + build: + context: .. + dockerfile: ./build/Dockerfile.dev + ports: + - "8080:8080" + - "40000:40000" + restart: always + depends_on: + - postgres-db + volumes: + - ./:/usr/src/app/ + +volumes: + api: + database_postgres: \ No newline at end of file diff --git a/examples/healthCheck.http b/examples/healthCheck.http new file mode 100644 index 0000000..1fe0a99 --- /dev/null +++ b/examples/healthCheck.http @@ -0,0 +1,4 @@ +GET http://localhost:8080/health-check +Accept: application/json + +### diff --git a/examples/token.http b/examples/token.http new file mode 100644 index 0000000..dab3c17 --- /dev/null +++ b/examples/token.http @@ -0,0 +1,9 @@ +POST http://localhost:8080/token +Content-Type: application/json + +{ + "username": "username", + "password": "password" +} + +### diff --git a/go.mod b/go.mod new file mode 100644 index 0000000..6c11ec8 --- /dev/null +++ b/go.mod @@ -0,0 +1,27 @@ +module github.com/loukaspe/platform-go-challenge + +go 1.19 + +require ( + github.com/golang-jwt/jwt/v4 v4.5.0 + github.com/gorilla/mux v1.8.0 + github.com/joho/godotenv v1.5.1 + github.com/pkoukk/tiktoken-go v0.1.7 + github.com/sashabaranov/go-openai v1.40.0 + github.com/sirupsen/logrus v1.9.0 + golang.org/x/crypto v0.8.0 + gorm.io/driver/postgres v1.5.0 + gorm.io/gorm v1.25.0 +) + +require ( + github.com/dlclark/regexp2 v1.10.0 // indirect + github.com/google/uuid v1.3.0 // indirect + github.com/jackc/pgpassfile v1.0.0 // indirect + github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect + github.com/jackc/pgx/v5 v5.3.0 // indirect + github.com/jinzhu/inflection v1.0.0 // indirect + github.com/jinzhu/now v1.1.5 // indirect + golang.org/x/sys v0.7.0 // indirect + golang.org/x/text v0.9.0 // indirect +) diff --git a/go.sum b/go.sum new file mode 100644 index 0000000..a618082 --- /dev/null +++ b/go.sum @@ -0,0 +1,97 @@ +github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/dlclark/regexp2 v1.10.0 h1:+/GIL799phkJqYW+3YbOd8LCcbHzT0Pbo8zl70MHsq0= +github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= +github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg= +github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= +github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= +github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= +github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= +github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= +github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= +github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk= +github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM= +github.com/jackc/pgx/v5 v5.3.0 h1:/NQi8KHMpKWHInxXesC8yD4DhkXPrVhmnwYkjp9AmBA= +github.com/jackc/pgx/v5 v5.3.0/go.mod h1:t3JDKnCBlYIc0ewLF0Q7B8MXmoIaBOZj/ic7iHozM/8= +github.com/jackc/puddle/v2 v2.2.0/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4= +github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= +github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= +github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= +github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= +github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= +github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= +github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= +github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= +github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/pkoukk/tiktoken-go v0.1.7 h1:qOBHXX4PHtvIvmOtyg1EeKlwFRiMKAcoMp4Q+bLQDmw= +github.com/pkoukk/tiktoken-go v0.1.7/go.mod h1:9NiV+i9mJKGj1rYOT+njbv+ZwA/zJxYdewGl6qVatpg= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= +github.com/sashabaranov/go-openai v1.40.0 h1:Peg9Iag5mUJtPW00aYatlsn97YML0iNULiLNe74iPrU= +github.com/sashabaranov/go-openai v1.40.0/go.mod h1:lj5b/K+zjTSFxVLijLSTDZuP7adOgerWeFyZLUhAKRg= +github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= +github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= +github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= +golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58= +golang.org/x/crypto v0.8.0 h1:pd9TJtTueMTVQXzk8E2XESSMQDj/U7OUu0PqJqPXQjQ= +golang.org/x/crypto v0.8.0/go.mod h1:mRqEX+O9/h5TFCrQhkgjo2yKi0yYA+9ecGkdQoHrywE= +golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= +golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= +golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.7.0 h1:3jlCCIQZPdOYu1h8BkNvLz8Kgwtae2cagcG/VamtZRU= +golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= +golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE= +golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= +golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= +gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gorm.io/driver/postgres v1.5.0 h1:u2FXTy14l45qc3UeCJ7QaAXZmZfDDv0YrthvmRq1l0U= +gorm.io/driver/postgres v1.5.0/go.mod h1:FUZXzO+5Uqg5zzwzv4KK49R8lvGIyscBOqYrtI1Ce9A= +gorm.io/gorm v1.24.7-0.20230306060331-85eaf9eeda11/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= +gorm.io/gorm v1.25.0 h1:+KtYtb2roDz14EQe4bla8CbQlmb9dN3VejSai3lprfU= +gorm.io/gorm v1.25.0/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= diff --git a/internal/core/domain/jwtClaims.go b/internal/core/domain/jwtClaims.go new file mode 100644 index 0000000..b6145fe --- /dev/null +++ b/internal/core/domain/jwtClaims.go @@ -0,0 +1,16 @@ +package domain + +import ( + "context" + "github.com/golang-jwt/jwt/v4" +) + +type JwtClaims struct { + *jwt.RegisteredClaims + UserInfo interface{} +} +type JwtClaimsInterface interface { + CreateToken(sub string, userInfo interface{}) (string, error) + GetClaimsFromToken(tokenString string) (jwt.MapClaims, error) + SetJWTClaimsContext(ctx context.Context, claims jwt.MapClaims) context.Context +} diff --git a/internal/core/domain/user.go b/internal/core/domain/user.go new file mode 100644 index 0000000..4dc5e38 --- /dev/null +++ b/internal/core/domain/user.go @@ -0,0 +1,23 @@ +package domain + +import "golang.org/x/crypto/bcrypt" + +type User struct { + Username string + Password string +} + +const HashCost = 10 + +func (user *User) CheckPassword(providedPassword string) error { + return bcrypt.CompareHashAndPassword([]byte(user.Password), []byte(providedPassword)) +} + +func (user *User) HashPassword() error { + bytes, err := bcrypt.GenerateFromPassword([]byte(user.Password), HashCost) + if err != nil { + return err + } + user.Password = string(bytes) + return nil +} diff --git a/internal/core/services/jwtService.go b/internal/core/services/jwtService.go new file mode 100644 index 0000000..2df54dc --- /dev/null +++ b/internal/core/services/jwtService.go @@ -0,0 +1,30 @@ +package services + +import ( + "github.com/golang-jwt/jwt/v4" + "github.com/loukaspe/platform-go-challenge/internal/core/domain" +) + +type JwtService struct { + jwtDomain domain.JwtClaimsInterface +} + +func NewJwtService(domain domain.JwtClaimsInterface) *JwtService { + return &JwtService{jwtDomain: domain} +} + +func (j *JwtService) CreateJwtTokenService(user domain.User) (string, error) { + tokenValue, err := j.jwtDomain.CreateToken(user.Username, user) + if err != nil { + return "", err + } + return tokenValue, nil +} + +func (j *JwtService) ClaimsFromJwtTokenService(token string) (jwt.MapClaims, error) { + claims, err := j.jwtDomain.GetClaimsFromToken(token) + if err != nil { + return nil, err + } + return claims, nil +} diff --git a/internal/handlers/healthCheckHandler.go b/internal/handlers/healthCheckHandler.go new file mode 100644 index 0000000..9495be0 --- /dev/null +++ b/internal/handlers/healthCheckHandler.go @@ -0,0 +1,57 @@ +package handlers + +import ( + "encoding/json" + "gorm.io/gorm" + "net/http" +) + +type HealthCheckHandler struct{} + +func NewHealthCheckHandler(db *gorm.DB) *HealthCheckHandler { + return &HealthCheckHandler{} +} + +// Response when we do health check +// swagger:model HealthCheckResponse +type HealthCheckResponse struct { + // message that all is OK or what is wrong + // + // Required: true + Message string `json:"message"` +} + +// swagger:operation GET /health-check healthCheck +// +// # Check for the health of the app +// +// --- +// +// Consumes: +// - application/json +// +// Produces: +// - application/json +// +// Schemes: +// - http +// - https +// +// responses: +// "200": +// description: OK +// schema: +// $ref: "#/definitions/HealthCheckResponse" +// "500": +// description: Error +// schema: +// $ref: "#/definitions/HealthCheckResponse" +func (handler *HealthCheckHandler) HealthCheckController(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + + response := HealthCheckResponse{} + + response.Message = "OK" + json.NewEncoder(w).Encode(response) + w.WriteHeader(http.StatusOK) +} diff --git a/internal/handlers/jwtClaimsHandler.go b/internal/handlers/jwtClaimsHandler.go new file mode 100644 index 0000000..8830953 --- /dev/null +++ b/internal/handlers/jwtClaimsHandler.go @@ -0,0 +1,145 @@ +package handlers + +import ( + "encoding/json" + "github.com/loukaspe/platform-go-challenge/internal/core/domain" + "github.com/loukaspe/platform-go-challenge/internal/core/services" + "github.com/loukaspe/platform-go-challenge/pkg/logger" + "net/http" +) + +type JwtClaimsHandler struct { + service *services.JwtService + logger logger.LoggerInterface +} +type JwtClaimsHandlerInterface interface { + UrlAddController(w http.ResponseWriter, r *http.Request) +} + +func NewJwtClaimsHandler( + service *services.JwtService, + logger logger.LoggerInterface, +) *JwtClaimsHandler { + return &JwtClaimsHandler{ + service: service, + logger: logger, + } +} + +// request for generating jwt token +// +// swagger:parameters jwtToken +type JwtRequest struct { + // in:body + // Required: true + Username string `json:"username"` + // in:body + // Required: true + Password string `json:"password"` +} + +// Response with jwtToken +// swagger:model JwtResponse +type JwtResponse struct { + // jwt token + // + // Required: false + Token string `json:"token"` + // possible error message + // + // Required: false + ErrorMessage string `json:"errorMessage,omitempty"` +} + +// swagger:operation POST /token jwtToken +// +// # Generates JWT token for authentication and authorization +// +// --- +// +// Consumes: +// - application/json +// +// Produces: +// - application/json +// +// Schemes: +// - http +// - https +// +// responses: +// "200": +// description: OK +// schema: +// $ref: "#/definitions/JwtResponse" +// "500": +// description: Error +// schema: +// $ref: "#/definitions/JwtResponse" +func (handler *JwtClaimsHandler) JwtTokenController(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + + var request JwtRequest + var response JwtResponse + + err := json.NewDecoder(r.Body).Decode(&request) + if err != nil { + handler.logger.Error("Error in creating jwt token - request decode", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed auth request" + + handler.JsonResponse(w, http.StatusInternalServerError, &response) + + return + } + + if request.Username == "" || request.Password == "" { + response.ErrorMessage = "empty username or password" + + handler.JsonResponse(w, http.StatusBadRequest, &response) + + return + } + + // This is where I would implement the login, but I did not have time + + domainUser := domain.User{ + Username: request.Username, + Password: request.Password, + } + + result, err := handler.service.CreateJwtTokenService(domainUser) + if err != nil { + response.ErrorMessage = "error during creation of the token" + + handler.JsonResponse(w, http.StatusInternalServerError, &response) + + return + } + + response.Token = result + handler.JsonResponse(w, http.StatusOK, &response) + + return +} + +func (handler *JwtClaimsHandler) JsonResponse( + w http.ResponseWriter, + statusCode int, + response *JwtResponse, +) { + w.WriteHeader(statusCode) + err := json.NewEncoder(w).Encode(response) + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + response.ErrorMessage = "error in adding user favourite asset - json response" + + handler.logger.Error("Error in adding user favourite asset - json response", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + } +} diff --git a/internal/handlers/middleware.go b/internal/handlers/middleware.go new file mode 100644 index 0000000..95a7e9e --- /dev/null +++ b/internal/handlers/middleware.go @@ -0,0 +1,37 @@ +package handlers + +import ( + "github.com/loukaspe/platform-go-challenge/internal/core/domain" + "net/http" + "strings" +) + +type AuthenticationMw struct { + claimsDomain domain.JwtClaimsInterface +} +type AuthenticationMechanismInterface interface { + AuthenticationMW(next http.Handler) http.Handler +} + +func NewAuthenticationMw(claims domain.JwtClaimsInterface) *AuthenticationMw { + return &AuthenticationMw{claimsDomain: claims} +} +func (a *AuthenticationMw) AuthenticationMW(next http.Handler) http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + authHeader := r.Header.Get("Authorization") + if !strings.HasPrefix(authHeader, "Bearer") { + http.Error(w, "Not Authorized", http.StatusUnauthorized) + return + } + + tokenString := strings.TrimPrefix(authHeader, "Bearer ") + + claims, err := a.claimsDomain.GetClaimsFromToken(tokenString) + if err != nil { + http.Error(w, err.Error(), http.StatusUnauthorized) + return + } + r = r.WithContext(a.claimsDomain.SetJWTClaimsContext(r.Context(), claims)) + next.ServeHTTP(w, r) + }) +} diff --git a/internal/repositories/userDao.go b/internal/repositories/userDao.go new file mode 100644 index 0000000..3a6118b --- /dev/null +++ b/internal/repositories/userDao.go @@ -0,0 +1,9 @@ +package repositories + +import "gorm.io/gorm" + +type User struct { + gorm.Model + Username string `gorm:"not null;"` + Password string `gorm:"not null;"` +} diff --git a/pkg/auth/authMech.go b/pkg/auth/authMech.go new file mode 100644 index 0000000..478c87d --- /dev/null +++ b/pkg/auth/authMech.go @@ -0,0 +1,62 @@ +package auth + +import ( + "context" + "fmt" + "github.com/golang-jwt/jwt/v4" + "github.com/loukaspe/platform-go-challenge/internal/core/domain" + "time" +) + +type claimskey int + +var claimsKey claimskey + +type AuthMechanism struct { + secret []byte + signingMethod string +} + +func NewAuthMechanism( + secret string, + signingMethod string, +) *AuthMechanism { + return &AuthMechanism{ + secret: []byte(secret), + signingMethod: signingMethod, + } +} +func (j *AuthMechanism) CreateToken(sub string, userInfo interface{}) (string, error) { + token := jwt.New(jwt.GetSigningMethod(j.signingMethod)) + expiration := time.Now().Add(time.Hour) + token.Claims = &domain.JwtClaims{ + &jwt.RegisteredClaims{ + ExpiresAt: jwt.NewNumericDate(expiration), + Subject: sub, + }, userInfo, + } + val, err := token.SignedString(j.secret) + if err != nil { + + return "", err + } + return val, nil +} +func (j *AuthMechanism) GetClaimsFromToken(tokenString string) (jwt.MapClaims, error) { + token, err := jwt.Parse(tokenString, func(token *jwt.Token) (interface{}, error) { + if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok { + return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"]) + } + return j.secret, nil + }) + if err != nil { + return nil, err + } + if claims, ok := token.Claims.(jwt.MapClaims); ok && token.Valid { + return claims, nil + } + return nil, err +} +func (j *AuthMechanism) SetJWTClaimsContext(ctx context.Context, claims jwt.MapClaims) context.Context { + return context.WithValue(ctx, claimsKey, claims) +} diff --git a/pkg/errors/errors.go b/pkg/errors/errors.go new file mode 100644 index 0000000..2cf0876 --- /dev/null +++ b/pkg/errors/errors.go @@ -0,0 +1,61 @@ +package apierrors + +type UserNotFoundErrorWrapper struct { + ReturnedStatusCode int + OriginalError error +} + +// Error the original error message remains as it is for logging reasons etc. +// and the wrapper error message is empty because we don't want the client to see anything +func (err UserNotFoundErrorWrapper) Error() string { + return "" +} + +func (err UserNotFoundErrorWrapper) Unwrap() error { + return err.OriginalError +} + +type AssetNotFoundErrorWrapper struct { + ReturnedStatusCode int + OriginalError error +} + +// Error the original error message remains as it is for logging reasons etc. +// and the wrapper error message is empty because we don't want the client to see anything +func (err AssetNotFoundErrorWrapper) Error() string { + return "" +} + +func (err AssetNotFoundErrorWrapper) Unwrap() error { + return err.OriginalError +} + +type UnknownAssetTypeErrorWrapper struct { + ReturnedStatusCode int + OriginalError error +} + +// Error the original error message remains as it is for logging reasons etc. +// and the wrapper error message is empty because we don't want the client to see anything +func (err UnknownAssetTypeErrorWrapper) Error() string { + return "" +} + +func (err UnknownAssetTypeErrorWrapper) Unwrap() error { + return err.OriginalError +} + +type NoFavouriteAssetsErrorWrapper struct { + ReturnedStatusCode int + OriginalError error +} + +// Error the original error message remains as it is for logging reasons etc. +// and the wrapper error message is empty because we don't want the client to see anything +func (err NoFavouriteAssetsErrorWrapper) Error() string { + return "" +} + +func (err NoFavouriteAssetsErrorWrapper) Unwrap() error { + return err.OriginalError +} diff --git a/pkg/logger/loggerInterface.go b/pkg/logger/loggerInterface.go new file mode 100644 index 0000000..21962ba --- /dev/null +++ b/pkg/logger/loggerInterface.go @@ -0,0 +1,9 @@ +package logger + +type LoggerInterface interface { + Debug(msg string, fields map[string]interface{}) + Info(msg string, fields map[string]interface{}) + Warn(msg string, fields map[string]interface{}) + Error(msg string, fields map[string]interface{}) + Fatal(msg string, fields map[string]interface{}) +} diff --git a/pkg/logger/logrus.go b/pkg/logger/logrus.go new file mode 100644 index 0000000..e47190c --- /dev/null +++ b/pkg/logger/logrus.go @@ -0,0 +1,39 @@ +package logger + +import ( + "context" + "github.com/sirupsen/logrus" + "os" +) + +type LogrusLogger struct { + logger *logrus.Logger + ctx context.Context +} + +func NewLogger(ctx context.Context) *LogrusLogger { + logger := logrus.New() + logger.Out = os.Stdout + + return &LogrusLogger{logger: logger, ctx: ctx} +} + +func (l *LogrusLogger) Debug(msg string, fields map[string]interface{}) { + l.logger.WithFields(fields).Debug(msg) +} + +func (l *LogrusLogger) Info(msg string, fields map[string]interface{}) { + l.logger.WithFields(fields).Info(msg) +} + +func (l *LogrusLogger) Warn(msg string, fields map[string]interface{}) { + l.logger.WithFields(fields).Warn(msg) +} + +func (l *LogrusLogger) Error(msg string, fields map[string]interface{}) { + l.logger.WithFields(fields).Error(msg) +} + +func (l *LogrusLogger) Fatal(msg string, fields map[string]interface{}) { + l.logger.WithFields(fields).Fatal(msg) +} diff --git a/pkg/server/routes.go b/pkg/server/routes.go new file mode 100644 index 0000000..3d328d1 --- /dev/null +++ b/pkg/server/routes.go @@ -0,0 +1,33 @@ +package server + +import ( + "github.com/loukaspe/platform-go-challenge/internal/core/services" + "github.com/loukaspe/platform-go-challenge/internal/handlers" + + "github.com/loukaspe/platform-go-challenge/pkg/auth" + "net/http" + "os" +) + +func (s *Server) initializeRoutes() { + // health check + healthCheckHandler := handlers.NewHealthCheckHandler() + s.router.HandleFunc("/health-check", healthCheckHandler.HealthCheckController).Methods("GET") + + // auth + jwtMechanism := auth.NewAuthMechanism( + os.Getenv("JWT_SECRET_KEY"), + os.Getenv("JWT_SIGNING_METHOD"), + ) + jwtService := services.NewJwtService(jwtMechanism) + jwtMiddleware := handlers.NewAuthenticationMw(jwtMechanism) + jwtHandler := handlers.NewJwtClaimsHandler(jwtService, s.logger) + + s.router.HandleFunc("/token", jwtHandler.JwtTokenController).Methods(http.MethodPost) + + protected := s.router.PathPrefix("/").Subrouter() + protected.Use(jwtMiddleware.AuthenticationMW) + + //protected.HandleFunc("/users/{user_id:[0-9]+}/favourites", createUserFavouriteHandler.AddUserFavouriteAssetController).Methods("POST") + +} diff --git a/pkg/server/server.go b/pkg/server/server.go new file mode 100644 index 0000000..1720493 --- /dev/null +++ b/pkg/server/server.go @@ -0,0 +1,65 @@ +package server + +import ( + "context" + "errors" + "github.com/gorilla/mux" + "github.com/loukaspe/platform-go-challenge/pkg/logger" + log "github.com/sirupsen/logrus" + "gorm.io/gorm" + "net/http" + "os" + "os/signal" + "syscall" + "time" +) + +type Server struct { + DB *gorm.DB + httpServer *http.Server + router *mux.Router + logger logger.LoggerInterface +} + +func NewServer( + db *gorm.DB, + router *mux.Router, + httpServer *http.Server, + logger logger.LoggerInterface, + +) *Server { + return &Server{ + DB: db, + router: router, + httpServer: httpServer, + logger: logger, + } +} + +func (s *Server) Run() { + s.initializeRoutes() + + go func() { + if err := s.httpServer.ListenAndServe(); err != nil && + !errors.Is(err, http.ErrServerClosed) { + log.Fatal(err) + } + }() + + quit := make(chan os.Signal, 1) + signal.Notify(quit, os.Interrupt, os.Kill, syscall.SIGTERM) + <-quit + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + if err := s.httpServer.Shutdown(ctx); err != nil { + log.Fatal(err) + } + db, err := s.DB.DB() + if err != nil { + log.Fatal(err) + } + err = db.Close() + if err != nil { + log.Fatal(err) + } +} diff --git a/script/Makefile b/script/Makefile new file mode 100644 index 0000000..44cb018 --- /dev/null +++ b/script/Makefile @@ -0,0 +1,89 @@ +generate-mock: +ifndef FILE + @echo "FILE parameter is missing" + @exit 1 +endif + make build-dev + @docker run --volume "$(PWD)"/../:/app --workdir /app \ + dev-build /bin/bash -c "mockgen -source=${FILE} -destination=mocks/mock_${FILE}" + +tests-unit: + make build-dev + @docker run \ + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build go test -short -cover -count=1 ./... + +tests-benchmark: + make build-dev + @docker run \ + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build go test ./... -bench=. + +tests-all: + make build-dev + @docker run \ + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build go test ./... -cover -count=1 + +tests-all-with-coverage: + make build-dev + @docker run \ + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build go test -count=1 -v -coverpkg=./... -coverprofile=profile.cov ./... ; go tool cover -func profile.cov + +run-linter: + make build-dev + @docker run \ + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build golangci-lint run --config ./config/.golangci.yml ./... + +generate-swagger-files: + make build-dev + @docker run \ + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build swagger generate spec -o ./docs/swagger.json --scan-models + +start-app: + docker-compose --file ../deployment/docker-compose.yml down + docker-compose --file ../deployment/docker-compose.yml up + +kill-app: + docker-compose --file ../deployment/docker-compose.yml down --remove-orphans --volumes + +rebuild-app: + docker-compose --file ../deployment/docker-compose.yml stop app + docker-compose --file ../deployment/docker-compose.yml build app + docker-compose --file ../deployment/docker-compose.yml up -d app + +build-dev: + @docker build \ + --tag dev-build \ + -f ../build/Dockerfile.utilities .. + +migrate-data-small: + @docker run \ + --rm \ + --volume "$(PWD)":/scripts \ + -e PGPASSWORD=password \ + postgres \ + psql -h 172.18.0.1 -p 5432 -U username -d mydb -w -f ./scripts/migrationFileSmall.sql + +migrate-data-large: + @docker run \ + --rm \ + --volume "$(PWD)":/scripts \ + -e PGPASSWORD=password \ + postgres \ + psql -h 172.18.0.1 -p 5432 -U username -d mydb -w -f ./scripts/migrationFileLarge.sql \ No newline at end of file From 7b5fe73c34053b6951f603525d45d4dd67224339 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Sun, 25 May 2025 15:54:37 +0300 Subject: [PATCH 03/52] Small fix --- cmd/http/main.go | 6 +- deployment/docker-compose.yml | 2 +- go.mod | 38 +++++++--- go.sum | 103 ++++++++++++-------------- internal/core/services/jwtService.go | 2 +- internal/handlers/jwtClaimsHandler.go | 6 +- internal/handlers/middleware.go | 2 +- pkg/auth/authMech.go | 2 +- pkg/server/routes.go | 6 +- pkg/server/server.go | 2 +- 10 files changed, 89 insertions(+), 80 deletions(-) diff --git a/cmd/http/main.go b/cmd/http/main.go index 6357b35..bab386c 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -5,9 +5,9 @@ import ( "fmt" "github.com/gorilla/mux" "github.com/joho/godotenv" - "github.com/loukaspe/platform-go-challenge/internal/repositories" - "github.com/loukaspe/platform-go-challenge/pkg/logger" - "github.com/loukaspe/platform-go-challenge/pkg/server" + "github.com/loukaspe/jedi-team-challenge/pkg/chunks" + "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" + "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" log "github.com/sirupsen/logrus" "gorm.io/driver/postgres" "gorm.io/gorm" diff --git a/deployment/docker-compose.yml b/deployment/docker-compose.yml index 13a7d04..cf18e1c 100755 --- a/deployment/docker-compose.yml +++ b/deployment/docker-compose.yml @@ -24,7 +24,7 @@ services: ports: - "5050:80" app: -# container_name: platform-go-challenge-app +# container_name: jedi-team-challenge-app build: context: .. dockerfile: ./build/Dockerfile.dev diff --git a/go.mod b/go.mod index 6c11ec8..dfcb078 100644 --- a/go.mod +++ b/go.mod @@ -1,27 +1,43 @@ -module github.com/loukaspe/platform-go-challenge +module github.com/loukaspe/jedi-team-challenge -go 1.19 +go 1.21 + +toolchain go1.23.1 require ( github.com/golang-jwt/jwt/v4 v4.5.0 github.com/gorilla/mux v1.8.0 github.com/joho/godotenv v1.5.1 + github.com/openai/openai-go v1.1.0 + github.com/pinecone-io/go-pinecone/v3 v3.1.0 github.com/pkoukk/tiktoken-go v0.1.7 - github.com/sashabaranov/go-openai v1.40.0 github.com/sirupsen/logrus v1.9.0 - golang.org/x/crypto v0.8.0 - gorm.io/driver/postgres v1.5.0 + golang.org/x/crypto v0.32.0 gorm.io/gorm v1.25.0 ) require ( + github.com/apapsch/go-jsonmerge/v2 v2.0.0 // indirect + github.com/davecgh/go-spew v1.1.1 // indirect github.com/dlclark/regexp2 v1.10.0 // indirect - github.com/google/uuid v1.3.0 // indirect - github.com/jackc/pgpassfile v1.0.0 // indirect - github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect - github.com/jackc/pgx/v5 v5.3.0 // indirect + github.com/google/uuid v1.6.0 // indirect github.com/jinzhu/inflection v1.0.0 // indirect github.com/jinzhu/now v1.1.5 // indirect - golang.org/x/sys v0.7.0 // indirect - golang.org/x/text v0.9.0 // indirect + github.com/kr/pretty v0.3.0 // indirect + github.com/oapi-codegen/runtime v1.1.1 // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + github.com/stretchr/testify v1.8.4 // indirect + github.com/tidwall/gjson v1.14.4 // indirect + github.com/tidwall/match v1.1.1 // indirect + github.com/tidwall/pretty v1.2.1 // indirect + github.com/tidwall/sjson v1.2.5 // indirect + golang.org/x/net v0.34.0 // indirect + golang.org/x/sys v0.30.0 // indirect + golang.org/x/text v0.22.0 // indirect + google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157 // indirect + google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157 // indirect + google.golang.org/grpc v1.65.0 // indirect + google.golang.org/protobuf v1.34.1 // indirect + gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c // indirect + gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index a618082..6bd07ba 100644 --- a/go.sum +++ b/go.sum @@ -1,3 +1,7 @@ +github.com/RaveNoX/go-jsoncommentstrip v1.0.0/go.mod h1:78ihd09MekBnJnxpICcwzCMzGrKSKYe4AqU6PDYYpjk= +github.com/apapsch/go-jsonmerge/v2 v2.0.0 h1:axGnT1gRIfimI7gJifB699GoE/oq+F2MU7Dml6nw9rQ= +github.com/apapsch/go-jsonmerge/v2 v2.0.0/go.mod h1:lvDnEdqiQrp0O42VQGgmlKpxL1AP2+08jFMw88y4klk= +github.com/bmatcuk/doublestar v1.1.1/go.mod h1:UD6OnuiIn0yFxxA2le/rnRU1G4RaI4UvFv1sNto9p6w= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= @@ -6,92 +10,81 @@ github.com/dlclark/regexp2 v1.10.0 h1:+/GIL799phkJqYW+3YbOd8LCcbHzT0Pbo8zl70MHsq github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg= github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= -github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= -github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= +github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= +github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= -github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= -github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= -github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk= -github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM= -github.com/jackc/pgx/v5 v5.3.0 h1:/NQi8KHMpKWHInxXesC8yD4DhkXPrVhmnwYkjp9AmBA= -github.com/jackc/pgx/v5 v5.3.0/go.mod h1:t3JDKnCBlYIc0ewLF0Q7B8MXmoIaBOZj/ic7iHozM/8= -github.com/jackc/puddle/v2 v2.2.0/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4= github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= +github.com/juju/gnuflag v0.0.0-20171113085948-2ce1bb71843d/go.mod h1:2PavIy+JPciBPrBUjwbNvtwB6RQlve+hkpll6QSNmOE= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= +github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= +github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/oapi-codegen/runtime v1.1.1 h1:EXLHh0DXIJnWhdRPN2w4MXAzFyE4CskzhNLUmtpMYro= +github.com/oapi-codegen/runtime v1.1.1/go.mod h1:SK9X900oXmPWilYR5/WKPzt3Kqxn/uS/+lbpREv+eCg= +github.com/openai/openai-go v1.1.0 h1:daSn+y+3QJUmLV1xfh7B8QtgJYRw1hg3yWxKtQDfROE= +github.com/openai/openai-go v1.1.0/go.mod h1:g461MYGXEXBVdV5SaR/5tNzNbSfwTBBefwc+LlDCK0Y= +github.com/pinecone-io/go-pinecone/v3 v3.1.0 h1:JxUK7OXycfqOF+DZbCexT5jKGVA8s5gswZL1wS95zf8= +github.com/pinecone-io/go-pinecone/v3 v3.1.0/go.mod h1:v8VJwwmZFesCP3bIYv98eU/kIpT7v8s0UulNTLWR8c8= github.com/pkoukk/tiktoken-go v0.1.7 h1:qOBHXX4PHtvIvmOtyg1EeKlwFRiMKAcoMp4Q+bLQDmw= github.com/pkoukk/tiktoken-go v0.1.7/go.mod h1:9NiV+i9mJKGj1rYOT+njbv+ZwA/zJxYdewGl6qVatpg= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/rogpeppe/go-internal v1.6.1 h1:/FiVV8dS/e+YqF2JvO3yXRFbBLTIuSDkuC7aBOAvL+k= github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= -github.com/sashabaranov/go-openai v1.40.0 h1:Peg9Iag5mUJtPW00aYatlsn97YML0iNULiLNe74iPrU= -github.com/sashabaranov/go-openai v1.40.0/go.mod h1:lj5b/K+zjTSFxVLijLSTDZuP7adOgerWeFyZLUhAKRg= github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +github.com/spkg/bom v0.0.0-20160624110644-59b7046e48ad/go.mod h1:qLr4V1qq6nMqFKkMo8ZTx3f+BZEkzsRUY10Xsm2mwU0= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= -github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= -github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= -github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= -github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= -github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= -golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= -golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= -golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58= -golang.org/x/crypto v0.8.0 h1:pd9TJtTueMTVQXzk8E2XESSMQDj/U7OUu0PqJqPXQjQ= -golang.org/x/crypto v0.8.0/go.mod h1:mRqEX+O9/h5TFCrQhkgjo2yKi0yYA+9ecGkdQoHrywE= -golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= -golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= -golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= -golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk= +github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= +github.com/tidwall/gjson v1.14.2/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= +github.com/tidwall/gjson v1.14.4 h1:uo0p8EbA09J7RQaflQ1aBRffTR7xedD2bcIVSYxLnkM= +github.com/tidwall/gjson v1.14.4/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= +github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA= +github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM= +github.com/tidwall/pretty v1.2.0/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= +github.com/tidwall/pretty v1.2.1 h1:qjsOFOWWQl+N3RsoF5/ssm1pHmJJwhjlSbZ51I6wMl4= +github.com/tidwall/pretty v1.2.1/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= +github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY= +github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28= +golang.org/x/crypto v0.32.0 h1:euUpcYgM8WcP71gNpTqQCn6rC2t6ULUPiOzfWaXVVfc= +golang.org/x/crypto v0.32.0/go.mod h1:ZnnJkOaASj8g0AjIduWNlq2NRxL0PlBrbKVyZ6V/Ugc= +golang.org/x/net v0.34.0 h1:Mb7Mrk043xzHgnRM88suvJFwzVrRfHEHJEl5/71CKw0= +golang.org/x/net v0.34.0/go.mod h1:di0qlW3YNM5oh6GqDGQr92MyTozJPmybPK4Ev/Gm31k= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.7.0 h1:3jlCCIQZPdOYu1h8BkNvLz8Kgwtae2cagcG/VamtZRU= -golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= -golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= -golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= -golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= -golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= -golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE= -golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= -golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= -golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= -golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc= +golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/text v0.22.0 h1:bofq7m3/HAFvbF51jz3Q9wLg3jkvSPuiZu/pD1XwgtM= +golang.org/x/text v0.22.0/go.mod h1:YRoo4H8PVmsu+E3Ou7cqLVH8oXWIHVoX0jqUWALQhfY= +google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157 h1:7whR9kGa5LUwFtpLm2ArCEejtnxlGeLbAyjFY8sGNFw= +google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157/go.mod h1:99sLkeliLXfdj2J75X3Ho+rrVCaJze0uwN7zDDkjPVU= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157 h1:Zy9XzmMEflZ/MAaA7vNcoebnRAld7FsPW1EeBB7V0m8= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157/go.mod h1:EfXuqaE1J41VCDicxHzUDm+8rk+7ZdXzHV0IhO/I6s0= +google.golang.org/grpc v1.65.0 h1:bs/cUb4lp1G5iImFFd3u5ixQzweKizoZJAwBNLR42lc= +google.golang.org/grpc v1.65.0/go.mod h1:WgYC2ypjlB0EiQi6wdKixMqukr6lBc0Vo+oOgjrM5ZQ= +google.golang.org/protobuf v1.34.1 h1:9ddQBjfCyZPOHPUiPxpYESBLc+T8P3E+Vo4IbKZgFWg= +google.golang.org/protobuf v1.34.1/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -gorm.io/driver/postgres v1.5.0 h1:u2FXTy14l45qc3UeCJ7QaAXZmZfDDv0YrthvmRq1l0U= -gorm.io/driver/postgres v1.5.0/go.mod h1:FUZXzO+5Uqg5zzwzv4KK49R8lvGIyscBOqYrtI1Ce9A= -gorm.io/gorm v1.24.7-0.20230306060331-85eaf9eeda11/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= gorm.io/gorm v1.25.0 h1:+KtYtb2roDz14EQe4bla8CbQlmb9dN3VejSai3lprfU= gorm.io/gorm v1.25.0/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= diff --git a/internal/core/services/jwtService.go b/internal/core/services/jwtService.go index 2df54dc..b9a0d97 100644 --- a/internal/core/services/jwtService.go +++ b/internal/core/services/jwtService.go @@ -2,7 +2,7 @@ package services import ( "github.com/golang-jwt/jwt/v4" - "github.com/loukaspe/platform-go-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" ) type JwtService struct { diff --git a/internal/handlers/jwtClaimsHandler.go b/internal/handlers/jwtClaimsHandler.go index 8830953..dc0746a 100644 --- a/internal/handlers/jwtClaimsHandler.go +++ b/internal/handlers/jwtClaimsHandler.go @@ -2,9 +2,9 @@ package handlers import ( "encoding/json" - "github.com/loukaspe/platform-go-challenge/internal/core/domain" - "github.com/loukaspe/platform-go-challenge/internal/core/services" - "github.com/loukaspe/platform-go-challenge/pkg/logger" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" "net/http" ) diff --git a/internal/handlers/middleware.go b/internal/handlers/middleware.go index 95a7e9e..80f7d1a 100644 --- a/internal/handlers/middleware.go +++ b/internal/handlers/middleware.go @@ -1,7 +1,7 @@ package handlers import ( - "github.com/loukaspe/platform-go-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "net/http" "strings" ) diff --git a/pkg/auth/authMech.go b/pkg/auth/authMech.go index 478c87d..1192484 100644 --- a/pkg/auth/authMech.go +++ b/pkg/auth/authMech.go @@ -4,7 +4,7 @@ import ( "context" "fmt" "github.com/golang-jwt/jwt/v4" - "github.com/loukaspe/platform-go-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "time" ) diff --git a/pkg/server/routes.go b/pkg/server/routes.go index 3d328d1..fc7ba53 100644 --- a/pkg/server/routes.go +++ b/pkg/server/routes.go @@ -1,10 +1,10 @@ package server import ( - "github.com/loukaspe/platform-go-challenge/internal/core/services" - "github.com/loukaspe/platform-go-challenge/internal/handlers" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + "github.com/loukaspe/jedi-team-challenge/internal/handlers" - "github.com/loukaspe/platform-go-challenge/pkg/auth" + "github.com/loukaspe/jedi-team-challenge/pkg/auth" "net/http" "os" ) diff --git a/pkg/server/server.go b/pkg/server/server.go index 1720493..9a715cc 100644 --- a/pkg/server/server.go +++ b/pkg/server/server.go @@ -4,7 +4,7 @@ import ( "context" "errors" "github.com/gorilla/mux" - "github.com/loukaspe/platform-go-challenge/pkg/logger" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" log "github.com/sirupsen/logrus" "gorm.io/gorm" "net/http" From 8bf4dad0cf7f3030622c8a2f949a1427e9155159 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Sun, 25 May 2025 15:54:47 +0300 Subject: [PATCH 04/52] Adds initial data embedding --- cmd/http/main.go | 124 ++++++++++++++++++++++++++++--------- pkg/chunks/chunker.go | 74 ++++++++++++++++++++++ pkg/embeddings/embedder.go | 70 +++++++++++++++++++++ pkg/vectordb/pinecone.go | 63 +++++++++++++++++++ 4 files changed, 301 insertions(+), 30 deletions(-) create mode 100644 pkg/chunks/chunker.go create mode 100644 pkg/embeddings/embedder.go create mode 100644 pkg/vectordb/pinecone.go diff --git a/cmd/http/main.go b/cmd/http/main.go index bab386c..eabb74e 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -3,60 +3,124 @@ package main import ( "context" "fmt" - "github.com/gorilla/mux" "github.com/joho/godotenv" "github.com/loukaspe/jedi-team-challenge/pkg/chunks" "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" + "github.com/openai/openai-go" + "github.com/openai/openai-go/option" + "github.com/pinecone-io/go-pinecone/v3/pinecone" + "github.com/pkoukk/tiktoken-go" log "github.com/sirupsen/logrus" - "gorm.io/driver/postgres" - "gorm.io/gorm" - "net/http" "os" + "strconv" ) func main() { getEnv() - logger := logger.NewLogger(context.Background()) - router := mux.NewRouter() - httpServer := &http.Server{ - Addr: os.Getenv("SERVER_ADDR"), - Handler: router, + pineconeAPIKey := os.Getenv("PINECONE_API_KEY") + pineconeIndexName := os.Getenv("PINECONE_INDEX") + openAIAPIKey := os.Getenv("OPENAI_API_KEY") + //openAIURL := os.Getenv("OPENAI_URL") + chunkEncoding := os.Getenv("CHUNK_ENCODING_MODEL") + embeddingModel := openai.EmbeddingModel(os.Getenv("EMBEDDING_MODEL")) + maxTokensPerChunksAsString := os.Getenv("MAX_TOKENS_PER_CHUNKS") + maxTokensPerChunks, err := strconv.Atoi(maxTokensPerChunksAsString) + if err != nil { + log.Fatal("Cannot read max token per chunks: ", err) } - db := getDB() - server := server.NewServer(db, router, httpServer, logger) + tiktokenEncoder, err := tiktoken.GetEncoding(chunkEncoding) + if err != nil { + log.Fatal("Cannot create encoder: ", err) + } - server.Run() -} + chunker, err := chunks.NewTiktokenChunker(tiktokenEncoder, maxTokensPerChunks) + if err != nil { + log.Fatal("Cannot create chunker: ", err) + } -func getDB() *gorm.DB { - dbDsn := fmt.Sprintf( - "host=%s port=%s user=%s dbname=%s sslmode=disable password=%s TimeZone=Europe/Athens", - os.Getenv("DB_HOST"), - os.Getenv("DB_PORT"), - os.Getenv("DB_USER"), - os.Getenv("DB_NAME"), - os.Getenv("DB_PASSWORD"), - ) - db, err := gorm.Open(postgres.Open(dbDsn), &gorm.Config{}) + openAIClient := openai.NewClient(option.WithAPIKey(openAIAPIKey)) + embedder := embeddings.NewEmbeddingService(&openAIClient, embeddingModel) + + textBytes, err := os.ReadFile("./data.md") if err != nil { - log.Fatal("Cannot connect to database: ", err) + log.Fatal(err) } + text := string(textBytes) - // Drops added in order to start with clean DB on App start for - // assessment reasons - db.Migrator().DropTable("users") + chunks := chunker.Chunk(text) + fmt.Printf("Generated %d chunks\n", len(chunks)) - err = db.AutoMigrate(&repositories.User{}) + ctx := context.Background() + embeddings, err := embedder.Embed(ctx, chunks) if err != nil { - log.Fatal("cannot migrate user table") + log.Fatalf("Embedding error: %v", err) + } + + // Output + for i, emb := range embeddings { + fmt.Printf("Chunk %d → %d-dim vector\n", i, len(emb)) } - return db + pineconeClient, err := pinecone.NewClient(pinecone.NewClientParams{ + ApiKey: pineconeAPIKey, + }) + if err != nil { + log.Fatalf("Failed to create pinecone Client: %v", err) + } + pineconeVectorDB := vectordb.NewPineconeVectorDB( + pineconeIndexName, + pineconeClient, + ) + + count, err := pineconeVectorDB.StoreEmbeddings(ctx, embeddings) + if err != nil { + log.Fatalf("Failed to store embeddings: %v", err) + } + + fmt.Sprintf("Stored %d embeddings in Pinecone index %s\n", count, pineconeIndexName) + + //logger := logger.NewLogger(context.Background()) + //router := mux.NewRouter() + //httpServer := &http.Server{ + // Addr: os.Getenv("SERVER_ADDR"), + // Handler: router, + //} + //db := getDB() + // + //server := server.NewServer(db, router, httpServer, logger) + // + //server.Run() } +//func getDB() *gorm.DB { +// dbDsn := fmt.Sprintf( +// "host=%s port=%s user=%s dbname=%s sslmode=disable password=%s TimeZone=Europe/Athens", +// os.Getenv("DB_HOST"), +// os.Getenv("DB_PORT"), +// os.Getenv("DB_USER"), +// os.Getenv("DB_NAME"), +// os.Getenv("DB_PASSWORD"), +// ) +// db, err := gorm.Open(postgres.Open(dbDsn), &gorm.Config{}) +// if err != nil { +// log.Fatal("Cannot connect to database: ", err) +// } +// +// // Drops added in order to start with clean DB on App start for +// // assessment reasons +// db.Migrator().DropTable("users") +// +// err = db.AutoMigrate(&repositories.User{}) +// if err != nil { +// log.Fatal("cannot migrate user table") +// } +// +// return db +//} + func getEnv() { err := godotenv.Load("./config/.env") if err != nil { diff --git a/pkg/chunks/chunker.go b/pkg/chunks/chunker.go new file mode 100644 index 0000000..873a8ab --- /dev/null +++ b/pkg/chunks/chunker.go @@ -0,0 +1,74 @@ +package chunks + +import ( + "github.com/pkoukk/tiktoken-go" + "strings" +) + +type TiktokenChunker struct { + Encoder *tiktoken.Tiktoken + MaxTokensPerChunk int +} + +// TiktokenEncoding string +// enc, err := tiktoken.GetEncoding(encName) +func NewTiktokenChunker(encoder *tiktoken.Tiktoken, maxTokens int) (*TiktokenChunker, error) { + return &TiktokenChunker{Encoder: encoder, MaxTokensPerChunk: maxTokens}, nil +} + +func (c *TiktokenChunker) Chunk(text string) []string { + var chunks []string + sentences := splitToSentences(text) + + var buf strings.Builder + var count int + + flush := func() { + s := strings.TrimSpace(buf.String()) + if s != "" { + chunks = append(chunks, s) + } + buf.Reset() + count = 0 + } + + for _, sent := range sentences { + tokCount := len(c.Encoder.Encode(sent, nil, nil)) + if tokCount > c.MaxTokensPerChunk { + + for _, w := range strings.Fields(sent) { + wTok := len(c.Encoder.Encode(w+" ", nil, nil)) + if count+wTok > c.MaxTokensPerChunk { + flush() + } + buf.WriteString(w + " ") + count += wTok + } + continue + } + if count+tokCount > c.MaxTokensPerChunk { + flush() + } + buf.WriteString(sent) + count += tokCount + } + flush() + return chunks +} + +func splitToSentences(text string) []string { + var sents []string + var sb strings.Builder + for _, ch := range text { + sb.WriteRune(ch) + if ch == '.' || ch == '!' || ch == '?' { + sents = append(sents, sb.String()) + sb.Reset() + } + } + // leftover + if rem := strings.TrimSpace(sb.String()); rem != "" { + sents = append(sents, rem) + } + return sents +} diff --git a/pkg/embeddings/embedder.go b/pkg/embeddings/embedder.go new file mode 100644 index 0000000..0698104 --- /dev/null +++ b/pkg/embeddings/embedder.go @@ -0,0 +1,70 @@ +package embeddings + +import ( + "context" + "errors" + "fmt" + "github.com/openai/openai-go" + "math" + "time" +) + +const maxEmbeddingRetries = 5 + +//type Chunker interface { +// Chunk(text string) []string +//} + +type EmbeddingService struct { + // TODO: exei nohma edw to interface + OpenAIClient *openai.Client + EmbeddingModel openai.EmbeddingModel +} + +func NewEmbeddingService(client *openai.Client, embeddingModel openai.EmbeddingModel) *EmbeddingService { + return &EmbeddingService{ + OpenAIClient: client, + EmbeddingModel: embeddingModel, + } +} + +func (s *EmbeddingService) Embed(ctx context.Context, inputs []string) ([][]float64, error) { + var resp *openai.CreateEmbeddingResponse + var err error + + for attempt := 0; attempt < maxEmbeddingRetries; attempt++ { + resp, err = s.OpenAIClient.Embeddings.New( + ctx, + openai.EmbeddingNewParams{ + Model: s.EmbeddingModel, + Input: openai.EmbeddingNewParamsInputUnion{OfArrayOfStrings: inputs}, + }, + ) + + if err == nil { + break + } + + // see if it's an APIError with HTTP 429 + var apiErr *openai.Error + if errors.As(err, &apiErr) && apiErr.StatusCode == 429 { + wait := time.Duration(math.Pow(2, float64(attempt))) * time.Second + // TODO: log + fmt.Printf("Rate‐limited on attempt %d: waiting %s before retry…", attempt+1, wait) + time.Sleep(wait) + continue + } + // some other error – give up immediately + return nil, fmt.Errorf("failed embedding: %w", err) + } + + if err != nil { + return nil, err + } + // Collect embeddings + embeddings := make([][]float64, len(resp.Data)) + for i, d := range resp.Data { + embeddings[i] = d.Embedding + } + return embeddings, nil +} diff --git a/pkg/vectordb/pinecone.go b/pkg/vectordb/pinecone.go new file mode 100644 index 0000000..5b61fc3 --- /dev/null +++ b/pkg/vectordb/pinecone.go @@ -0,0 +1,63 @@ +package vectordb + +import ( + "context" + "fmt" + "github.com/pinecone-io/go-pinecone/v3/pinecone" +) + +type PineconeVectorDB struct { + client *pinecone.Client + index string +} + +func NewPineconeVectorDB(index string, client *pinecone.Client) *PineconeVectorDB { + return &PineconeVectorDB{ + index: index, + client: client, + } +} + +func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings [][]float64) (int, error) { + idx, err := db.client.DescribeIndex(ctx, db.index) + if err != nil { + return 0, err + } + + idxConnection, err := db.client.Index(pinecone.NewIndexConnParams{Host: idx.Host}) + if err != nil { + return 0, err + } + + vectors := make([]*pinecone.Vector, len(embeddings)) + for i, vec := range embeddings { + id := fmt.Sprintf("doc1-chunk-%d", i) + + //md, err := structpb.NewStruct(map[string]interface{}{ + // "text": chunks[i], + //}) + //if err != nil { + // log.Fatalf("failed to create metadata struct: %v", err) + //} + + vectorToFloat32 := make([]float32, len(vec)) + for i, v := range vec { + vectorToFloat32[i] = float32(v) + } + + vectors[i] = &pinecone.Vector{ + Id: id, + Values: &vectorToFloat32, + //Metadata: map[string]interface{}{ + // "text": chunks[i], // store the original chunk if you want + //}, + } + } + + count, err := idxConnection.UpsertVectors(ctx, vectors) + if err != nil { + return 0, err + } + + return int(count), nil +} From b6d57477cde57e02ed0f93e97c818fc0f6d05592 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Mon, 26 May 2025 17:24:00 +0300 Subject: [PATCH 05/52] configuration changes --- build/{Dockerfile.dev => Dev.Dockerfile} | 4 +- build/Dockerfile | 2 +- ...kerfile.utilities => Utilities.Dockerfile} | 4 +- deployment/docker-compose.yml | 2 +- docs/swagger/docs.go | 146 ++++++++++++++++++ docs/swagger/swagger.json | 122 +++++++++++++++ docs/swagger/swagger.yaml | 80 ++++++++++ go.mod | 17 +- go.sum | 20 ++- script/Makefile | 22 ++- 10 files changed, 399 insertions(+), 20 deletions(-) rename build/{Dockerfile.dev => Dev.Dockerfile} (88%) rename build/{Dockerfile.utilities => Utilities.Dockerfile} (83%) create mode 100644 docs/swagger/docs.go create mode 100644 docs/swagger/swagger.json create mode 100644 docs/swagger/swagger.yaml diff --git a/build/Dockerfile.dev b/build/Dev.Dockerfile similarity index 88% rename from build/Dockerfile.dev rename to build/Dev.Dockerfile index a44de73..04a525f 100755 --- a/build/Dockerfile.dev +++ b/build/Dev.Dockerfile @@ -1,4 +1,4 @@ -FROM golang:1.19.3-alpine3.16 as builder +FROM golang:1.24.3-alpine3.21 as builder WORKDIR /app @@ -7,7 +7,7 @@ RUN go mod download RUN go install github.com/joho/godotenv/cmd/godotenv@v1.4.0 RUN go install github.com/go-delve/delve/cmd/dlv@latest -FROM golang:1.19.3-alpine3.16 +FROM golang:1.24.3-alpine3.21 RUN apk update RUN apk add build-base bash diff --git a/build/Dockerfile b/build/Dockerfile index 957c1b6..1054377 100755 --- a/build/Dockerfile +++ b/build/Dockerfile @@ -1,5 +1,5 @@ # Start from golang base image -FROM golang:1.19.3-alpine3.16 as builder +FROM golang:1.24.3-alpine3.21 as builder # Install git. # Git is required for fetching the dependencies. diff --git a/build/Dockerfile.utilities b/build/Utilities.Dockerfile similarity index 83% rename from build/Dockerfile.utilities rename to build/Utilities.Dockerfile index ae6f5b4..ac2a02f 100755 --- a/build/Dockerfile.utilities +++ b/build/Utilities.Dockerfile @@ -1,4 +1,4 @@ -FROM golang:1.19.3-alpine3.16 +FROM golang:1.24.3-alpine3.21 # Install git @@ -16,4 +16,4 @@ COPY .. . RUN go install github.com/golang/mock/mockgen@v1.6.0 RUN go install github.com/joho/godotenv/cmd/godotenv@v1.4.0 RUN wget -O- -nv https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b $(go env GOPATH)/bin v1.50.1 -RUN go install github.com/go-swagger/go-swagger/cmd/swagger@v0.30.4 \ No newline at end of file +RUN go install github.com/swaggo/swag/cmd/swag@latest \ No newline at end of file diff --git a/deployment/docker-compose.yml b/deployment/docker-compose.yml index cf18e1c..37b2474 100755 --- a/deployment/docker-compose.yml +++ b/deployment/docker-compose.yml @@ -27,7 +27,7 @@ services: # container_name: jedi-team-challenge-app build: context: .. - dockerfile: ./build/Dockerfile.dev + dockerfile: ./build/Dev.Dockerfile ports: - "8080:8080" - "40000:40000" diff --git a/docs/swagger/docs.go b/docs/swagger/docs.go new file mode 100644 index 0000000..2c41e12 --- /dev/null +++ b/docs/swagger/docs.go @@ -0,0 +1,146 @@ +// Package swagger Code generated by swaggo/swag. DO NOT EDIT +package swagger + +import "github.com/swaggo/swag" + +const docTemplate = `{ + "schemes": {{ marshal .Schemes }}, + "consumes": [ + "application/json" + ], + "produces": [ + "application/json" + ], + "swagger": "2.0", + "info": { + "description": "{{escape .Description}}", + "title": "{{.Title}}", + "contact": { + "name": "Loukas Peteinaris", + "url": "loukas.peteinaris@gmail.com" + }, + "version": "{{.Version}}" + }, + "host": "{{.Host}}", + "basePath": "{{.BasePath}}", + "paths": { + "/users/user_id/chat-sessions": { + "post": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Creates a chat session for User", + "summary": "Creates chat session", + "parameters": [ + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + } + } + } + } + }, + "definitions": { + "chatSessions.ChatSessionResponse": { + "type": "object", + "properties": { + "createdAt": { + "type": "string" + }, + "errorMessage": { + "type": "string" + }, + "id": { + "type": "string" + }, + "messages": { + "type": "array", + "items": { + "$ref": "#/definitions/chatSessions.Message" + } + }, + "title": { + "type": "string" + }, + "updatedAt": { + "type": "string" + } + } + }, + "chatSessions.Message": { + "type": "object", + "properties": { + "content": { + "type": "string" + }, + "id": { + "type": "string" + }, + "sender": { + "type": "string", + "example": "USER" + }, + "timestamp": { + "type": "string" + } + } + } + }, + "securityDefinitions": { + "BearerAuth": { + "description": "Header value should be in the form of ` + "`" + `Bearer \u003cJWT access token\u003e` + "`" + `", + "type": "apiKey", + "name": "Authorization", + "in": "header" + } + } +}` + +// SwaggerInfo holds exported Swagger Info so clients can modify it +var SwaggerInfo = &swag.Spec{ + Version: "1.0", + Host: "localhost:8080", + BasePath: "/", + Schemes: []string{}, + Title: "Louk Chatwalker", + Description: "GWI's Jedi Team Challenge", + InfoInstanceName: "swagger", + SwaggerTemplate: docTemplate, + LeftDelim: "{{", + RightDelim: "}}", +} + +func init() { + swag.Register(SwaggerInfo.InstanceName(), SwaggerInfo) +} diff --git a/docs/swagger/swagger.json b/docs/swagger/swagger.json new file mode 100644 index 0000000..9e27086 --- /dev/null +++ b/docs/swagger/swagger.json @@ -0,0 +1,122 @@ +{ + "consumes": [ + "application/json" + ], + "produces": [ + "application/json" + ], + "swagger": "2.0", + "info": { + "description": "GWI's Jedi Team Challenge", + "title": "Louk Chatwalker", + "contact": { + "name": "Loukas Peteinaris", + "url": "loukas.peteinaris@gmail.com" + }, + "version": "1.0" + }, + "host": "localhost:8080", + "basePath": "/", + "paths": { + "/users/user_id/chat-sessions": { + "post": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Creates a chat session for User", + "summary": "Creates chat session", + "parameters": [ + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + } + } + } + } + }, + "definitions": { + "chatSessions.ChatSessionResponse": { + "type": "object", + "properties": { + "createdAt": { + "type": "string" + }, + "errorMessage": { + "type": "string" + }, + "id": { + "type": "string" + }, + "messages": { + "type": "array", + "items": { + "$ref": "#/definitions/chatSessions.Message" + } + }, + "title": { + "type": "string" + }, + "updatedAt": { + "type": "string" + } + } + }, + "chatSessions.Message": { + "type": "object", + "properties": { + "content": { + "type": "string" + }, + "id": { + "type": "string" + }, + "sender": { + "type": "string", + "example": "USER" + }, + "timestamp": { + "type": "string" + } + } + } + }, + "securityDefinitions": { + "BearerAuth": { + "description": "Header value should be in the form of `Bearer \u003cJWT access token\u003e`", + "type": "apiKey", + "name": "Authorization", + "in": "header" + } + } +} \ No newline at end of file diff --git a/docs/swagger/swagger.yaml b/docs/swagger/swagger.yaml new file mode 100644 index 0000000..03fd6fb --- /dev/null +++ b/docs/swagger/swagger.yaml @@ -0,0 +1,80 @@ +basePath: / +consumes: +- application/json +definitions: + chatSessions.ChatSessionResponse: + properties: + createdAt: + type: string + errorMessage: + type: string + id: + type: string + messages: + items: + $ref: '#/definitions/chatSessions.Message' + type: array + title: + type: string + updatedAt: + type: string + type: object + chatSessions.Message: + properties: + content: + type: string + id: + type: string + sender: + example: USER + type: string + timestamp: + type: string + type: object +host: localhost:8080 +info: + contact: + name: Loukas Peteinaris + url: loukas.peteinaris@gmail.com + description: GWI's Jedi Team Challenge + title: Louk Chatwalker + version: "1.0" +paths: + /users/user_id/chat-sessions: + post: + description: Creates a chat session for User + parameters: + - description: user id + in: path + name: user_id + required: true + type: integer + responses: + "201": + description: Created + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + "400": + description: Error in message payload + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + "401": + description: Authentication error + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + "500": + description: Internal Server Error + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + security: + - BearerAuth: [] + summary: Creates chat session +produces: +- application/json +securityDefinitions: + BearerAuth: + description: Header value should be in the form of `Bearer ` + in: header + name: Authorization + type: apiKey +swagger: "2.0" diff --git a/go.mod b/go.mod index dfcb078..2c94034 100644 --- a/go.mod +++ b/go.mod @@ -5,39 +5,46 @@ go 1.21 toolchain go1.23.1 require ( + github.com/DATA-DOG/go-sqlmock v1.5.2 github.com/golang-jwt/jwt/v4 v4.5.0 + github.com/google/uuid v1.6.0 github.com/gorilla/mux v1.8.0 github.com/joho/godotenv v1.5.1 github.com/openai/openai-go v1.1.0 github.com/pinecone-io/go-pinecone/v3 v3.1.0 github.com/pkoukk/tiktoken-go v0.1.7 github.com/sirupsen/logrus v1.9.0 + github.com/stretchr/testify v1.8.4 golang.org/x/crypto v0.32.0 - gorm.io/gorm v1.25.0 + gorm.io/driver/postgres v1.5.11 + gorm.io/gorm v1.25.10 ) require ( github.com/apapsch/go-jsonmerge/v2 v2.0.0 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/dlclark/regexp2 v1.10.0 // indirect - github.com/google/uuid v1.6.0 // indirect + github.com/jackc/pgpassfile v1.0.0 // indirect + github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect + github.com/jackc/pgx/v5 v5.5.5 // indirect + github.com/jackc/puddle/v2 v2.2.1 // indirect github.com/jinzhu/inflection v1.0.0 // indirect github.com/jinzhu/now v1.1.5 // indirect - github.com/kr/pretty v0.3.0 // indirect + github.com/kr/text v0.2.0 // indirect github.com/oapi-codegen/runtime v1.1.1 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect - github.com/stretchr/testify v1.8.4 // indirect + github.com/rogpeppe/go-internal v1.6.1 // indirect github.com/tidwall/gjson v1.14.4 // indirect github.com/tidwall/match v1.1.1 // indirect github.com/tidwall/pretty v1.2.1 // indirect github.com/tidwall/sjson v1.2.5 // indirect golang.org/x/net v0.34.0 // indirect + golang.org/x/sync v0.11.0 // indirect golang.org/x/sys v0.30.0 // indirect golang.org/x/text v0.22.0 // indirect google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157 // indirect google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157 // indirect google.golang.org/grpc v1.65.0 // indirect google.golang.org/protobuf v1.34.1 // indirect - gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index 6bd07ba..11029cf 100644 --- a/go.sum +++ b/go.sum @@ -1,3 +1,5 @@ +github.com/DATA-DOG/go-sqlmock v1.5.2 h1:OcvFkGmslmlZibjAjaHm3L//6LiuBgolP7OputlJIzU= +github.com/DATA-DOG/go-sqlmock v1.5.2/go.mod h1:88MAG/4G7SMwSE3CeA0ZKzrT5CiOU3OJ+JlNzwDqpNU= github.com/RaveNoX/go-jsoncommentstrip v1.0.0/go.mod h1:78ihd09MekBnJnxpICcwzCMzGrKSKYe4AqU6PDYYpjk= github.com/apapsch/go-jsonmerge/v2 v2.0.0 h1:axGnT1gRIfimI7gJifB699GoE/oq+F2MU7Dml6nw9rQ= github.com/apapsch/go-jsonmerge/v2 v2.0.0/go.mod h1:lvDnEdqiQrp0O42VQGgmlKpxL1AP2+08jFMw88y4klk= @@ -16,6 +18,14 @@ github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= +github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= +github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= +github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk= +github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM= +github.com/jackc/pgx/v5 v5.5.5 h1:amBjrZVmksIdNjxGW/IiIMzxMKZFelXbUoPNb+8sjQw= +github.com/jackc/pgx/v5 v5.5.5/go.mod h1:ez9gk+OAat140fv9ErkZDYFWmXLfV+++K0uAOiwgm1A= +github.com/jackc/puddle/v2 v2.2.1 h1:RhxXJtFG022u4ibrCSMSiu5aOq1i77R3OHKNJj77OAk= +github.com/jackc/puddle/v2 v2.2.1/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4= github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= @@ -23,8 +33,8 @@ github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/ github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= github.com/juju/gnuflag v0.0.0-20171113085948-2ce1bb71843d/go.mod h1:2PavIy+JPciBPrBUjwbNvtwB6RQlve+hkpll6QSNmOE= +github.com/kisielk/sqlstruct v0.0.0-20201105191214-5f3e10d3ab46/go.mod h1:yyMNCyc/Ib3bDTKd379tNMpB/7/H5TjM2Y9QJ5THLbE= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= @@ -65,6 +75,8 @@ golang.org/x/crypto v0.32.0 h1:euUpcYgM8WcP71gNpTqQCn6rC2t6ULUPiOzfWaXVVfc= golang.org/x/crypto v0.32.0/go.mod h1:ZnnJkOaASj8g0AjIduWNlq2NRxL0PlBrbKVyZ6V/Ugc= golang.org/x/net v0.34.0 h1:Mb7Mrk043xzHgnRM88suvJFwzVrRfHEHJEl5/71CKw0= golang.org/x/net v0.34.0/go.mod h1:di0qlW3YNM5oh6GqDGQr92MyTozJPmybPK4Ev/Gm31k= +golang.org/x/sync v0.11.0 h1:GGz8+XQP4FvTTrjZPzNKTMFtSXH80RAzG+5ghFPgK9w= +golang.org/x/sync v0.11.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc= golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= @@ -86,5 +98,7 @@ gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -gorm.io/gorm v1.25.0 h1:+KtYtb2roDz14EQe4bla8CbQlmb9dN3VejSai3lprfU= -gorm.io/gorm v1.25.0/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= +gorm.io/driver/postgres v1.5.11 h1:ubBVAfbKEUld/twyKZ0IYn9rSQh448EdelLYk9Mv314= +gorm.io/driver/postgres v1.5.11/go.mod h1:DX3GReXH+3FPWGrrgffdvCk3DQ1dwDPdmbenSkweRGI= +gorm.io/gorm v1.25.10 h1:dQpO+33KalOA+aFYGlK+EfxcI5MbO7EP2yYygwh9h+s= +gorm.io/gorm v1.25.10/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= diff --git a/script/Makefile b/script/Makefile index 44cb018..bfc0ace 100644 --- a/script/Makefile +++ b/script/Makefile @@ -47,13 +47,23 @@ run-linter: --workdir /app \ dev-build golangci-lint run --config ./config/.golangci.yml ./... -generate-swagger-files: +swag-fmt: +# swag fmt -g ../../pkg/server/routes.go -d ../internal/handlers make build-dev @docker run \ - --rm \ - --volume "$(PWD)"/../:/app \ - --workdir /app \ - dev-build swagger generate spec -o ./docs/swagger.json --scan-models + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build swag fmt -g ../../pkg/server/routes.go -d ./internal/handlers + +swag: swag-fmt + #swag init -g ../../pkg/server/routes.go -d ../internal/handlers -o ../docs/swagger --parseDependency --parseDepth 1 + make build-dev + @docker run \ + --rm \ + --volume "$(PWD)"/../:/app \ + --workdir /app \ + dev-build swag init -g ../../pkg/server/routes.go -d ./internal/handlers -o ./docs/swagger --parseDependency --parseDepth 1 start-app: docker-compose --file ../deployment/docker-compose.yml down @@ -70,7 +80,7 @@ rebuild-app: build-dev: @docker build \ --tag dev-build \ - -f ../build/Dockerfile.utilities .. + -f ../build/Utilities.Dockerfile .. migrate-data-small: @docker run \ From e668e16586c5d4939e356b3f35c43d74c4967678 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Mon, 26 May 2025 17:52:58 +0300 Subject: [PATCH 06/52] Adds create chat session --- build/Utilities.Dockerfile | 2 +- cmd/http/main.go | 210 ++++++++------ go.mod | 19 +- go.sum | 71 ++++- internal/core/domain/chatSession.go | 23 ++ .../ports/chatSessionRepositoryInterface.go | 14 + internal/core/services/chatSessionService.go | 47 ++++ .../handlers/chatSessions/createHandler.go | 133 +++++++++ .../chatSessions/createHandler_test.go | 258 ++++++++++++++++++ internal/handlers/chatSessions/dto.go | 34 +++ internal/repositories/chatSessionDao.go | 23 ++ .../chatSessionRepositoryCreate.go | 36 +++ .../chatSessionRepositoryCreate_test.go | 136 +++++++++ internal/repositories/userDao.go | 14 +- .../core/services/chatSessionService.go | 100 +++++++ pkg/chunks/chunker.go | 8 +- pkg/errors/errors.go | 54 +--- pkg/server/routes.go | 28 +- 18 files changed, 1063 insertions(+), 147 deletions(-) create mode 100644 internal/core/domain/chatSession.go create mode 100644 internal/core/ports/chatSessionRepositoryInterface.go create mode 100644 internal/core/services/chatSessionService.go create mode 100644 internal/handlers/chatSessions/createHandler.go create mode 100644 internal/handlers/chatSessions/createHandler_test.go create mode 100644 internal/handlers/chatSessions/dto.go create mode 100644 internal/repositories/chatSessionDao.go create mode 100644 internal/repositories/chatSessionRepositoryCreate.go create mode 100644 internal/repositories/chatSessionRepositoryCreate_test.go create mode 100644 mocks/mock_internal/core/services/chatSessionService.go diff --git a/build/Utilities.Dockerfile b/build/Utilities.Dockerfile index ac2a02f..f50fe94 100755 --- a/build/Utilities.Dockerfile +++ b/build/Utilities.Dockerfile @@ -13,7 +13,7 @@ RUN go mod download # Copy the source from the current directory to the working Directory inside the container COPY .. . -RUN go install github.com/golang/mock/mockgen@v1.6.0 +RUN go install go.uber.org/mock/mockgen@latest RUN go install github.com/joho/godotenv/cmd/godotenv@v1.4.0 RUN wget -O- -nv https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b $(go env GOPATH)/bin v1.50.1 RUN go install github.com/swaggo/swag/cmd/swag@latest \ No newline at end of file diff --git a/cmd/http/main.go b/cmd/http/main.go index eabb74e..0a50736 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -3,15 +3,21 @@ package main import ( "context" "fmt" + "github.com/google/uuid" + "github.com/gorilla/mux" "github.com/joho/godotenv" + "github.com/loukaspe/jedi-team-challenge/internal/repositories" "github.com/loukaspe/jedi-team-challenge/pkg/chunks" "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" - "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/loukaspe/jedi-team-challenge/pkg/server" "github.com/openai/openai-go" "github.com/openai/openai-go/option" - "github.com/pinecone-io/go-pinecone/v3/pinecone" "github.com/pkoukk/tiktoken-go" log "github.com/sirupsen/logrus" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "net/http" "os" "strconv" ) @@ -19,111 +25,153 @@ import ( func main() { getEnv() - pineconeAPIKey := os.Getenv("PINECONE_API_KEY") - pineconeIndexName := os.Getenv("PINECONE_INDEX") - openAIAPIKey := os.Getenv("OPENAI_API_KEY") - //openAIURL := os.Getenv("OPENAI_URL") - chunkEncoding := os.Getenv("CHUNK_ENCODING_MODEL") - embeddingModel := openai.EmbeddingModel(os.Getenv("EMBEDDING_MODEL")) - maxTokensPerChunksAsString := os.Getenv("MAX_TOKENS_PER_CHUNKS") - maxTokensPerChunks, err := strconv.Atoi(maxTokensPerChunksAsString) - if err != nil { - log.Fatal("Cannot read max token per chunks: ", err) + encoder := getEncoder() + client := getOpenAIClient() + chunker := getChunker(encoder) + embedder := getEmbedder(&client) + + inputKnowledgeBase(chunker, embedder) + + logger := logger.NewLogger(context.Background()) + router := mux.NewRouter() + httpServer := &http.Server{ + Addr: os.Getenv("SERVER_ADDR"), + Handler: router, } + db := getDB() - tiktokenEncoder, err := tiktoken.GetEncoding(chunkEncoding) + server := server.NewServer(db, router, httpServer, logger) + + server.Run() +} + +func getDB() *gorm.DB { + dbDsn := fmt.Sprintf( + "host=%s port=%s user=%s dbname=%s sslmode=disable password=%s TimeZone=Europe/Athens", + os.Getenv("DB_HOST"), + os.Getenv("DB_PORT"), + os.Getenv("DB_USER"), + os.Getenv("DB_NAME"), + os.Getenv("DB_PASSWORD"), + ) + db, err := gorm.Open(postgres.Open(dbDsn), &gorm.Config{}) if err != nil { - log.Fatal("Cannot create encoder: ", err) + log.Fatal("Cannot connect to database: ", err) + } + + // Extension for UUID autogeneration as primary keys of tables + if err := db.Exec(`CREATE EXTENSION IF NOT EXISTS "uuid-ossp";`).Error; err != nil { + log.Fatal("failed to create uuid-ossp extension:", err) } - chunker, err := chunks.NewTiktokenChunker(tiktokenEncoder, maxTokensPerChunks) + // Drops added in order to start with clean DB on App start for + // assessment reasons + db.Migrator().DropTable("users") + db.Migrator().DropTable("chat_sessions") + + err = db.AutoMigrate(&repositories.User{}) if err != nil { - log.Fatal("Cannot create chunker: ", err) + log.Fatal("cannot migrate user table") } - openAIClient := openai.NewClient(option.WithAPIKey(openAIAPIKey)) - embedder := embeddings.NewEmbeddingService(&openAIClient, embeddingModel) + // TODO: remove + admin := repositories.User{ + ID: uuid.New(), + Username: "loukas", + Password: "loukastest", + } + fmt.Printf("Seeding user: %s\n", admin.ID) + err = db.Debug().Model(&repositories.User{}).Create(&admin).Error + if err != nil { + log.Fatalf("cannot seed users table: %v", err) + } - textBytes, err := os.ReadFile("./data.md") + err = db.AutoMigrate(&repositories.ChatSession{}) if err != nil { - log.Fatal(err) + log.Fatal("cannot migrate chat sessions table") } - text := string(textBytes) - chunks := chunker.Chunk(text) - fmt.Printf("Generated %d chunks\n", len(chunks)) + return db +} - ctx := context.Background() - embeddings, err := embedder.Embed(ctx, chunks) +func getEnv() { + err := godotenv.Load("./config/.env") if err != nil { - log.Fatalf("Embedding error: %v", err) + log.Fatalf("Error getting env, not comming through %v", err) } +} + +func getEncoder() *tiktoken.Tiktoken { + chunkEncoding := os.Getenv("CHUNK_ENCODING_MODEL") - // Output - for i, emb := range embeddings { - fmt.Printf("Chunk %d → %d-dim vector\n", i, len(emb)) + tiktokenEncoder, err := tiktoken.GetEncoding(chunkEncoding) + if err != nil { + log.Fatal("Cannot create encoder: ", err) } - pineconeClient, err := pinecone.NewClient(pinecone.NewClientParams{ - ApiKey: pineconeAPIKey, - }) + return tiktokenEncoder +} + +func getChunker(encoder *tiktoken.Tiktoken) *chunks.Chunker { + maxTokensPerChunksAsString := os.Getenv("MAX_TOKENS_PER_CHUNKS") + maxTokensPerChunks, err := strconv.Atoi(maxTokensPerChunksAsString) if err != nil { - log.Fatalf("Failed to create pinecone Client: %v", err) + log.Fatal("Cannot read max token per chunks: ", err) } - pineconeVectorDB := vectordb.NewPineconeVectorDB( - pineconeIndexName, - pineconeClient, - ) - count, err := pineconeVectorDB.StoreEmbeddings(ctx, embeddings) + chunker, err := chunks.NewChunker(encoder, maxTokensPerChunks) if err != nil { - log.Fatalf("Failed to store embeddings: %v", err) + log.Fatal("Cannot create chunker: ", err) } - fmt.Sprintf("Stored %d embeddings in Pinecone index %s\n", count, pineconeIndexName) + return chunker +} - //logger := logger.NewLogger(context.Background()) - //router := mux.NewRouter() - //httpServer := &http.Server{ - // Addr: os.Getenv("SERVER_ADDR"), - // Handler: router, - //} - //db := getDB() - // - //server := server.NewServer(db, router, httpServer, logger) - // - //server.Run() +func getEmbedder(client *openai.Client) *embeddings.EmbeddingService { + return embeddings.NewEmbeddingService(client, openai.EmbeddingModel(os.Getenv("EMBEDDING_MODEL"))) } -//func getDB() *gorm.DB { -// dbDsn := fmt.Sprintf( -// "host=%s port=%s user=%s dbname=%s sslmode=disable password=%s TimeZone=Europe/Athens", -// os.Getenv("DB_HOST"), -// os.Getenv("DB_PORT"), -// os.Getenv("DB_USER"), -// os.Getenv("DB_NAME"), -// os.Getenv("DB_PASSWORD"), -// ) -// db, err := gorm.Open(postgres.Open(dbDsn), &gorm.Config{}) -// if err != nil { -// log.Fatal("Cannot connect to database: ", err) -// } -// -// // Drops added in order to start with clean DB on App start for -// // assessment reasons -// db.Migrator().DropTable("users") -// -// err = db.AutoMigrate(&repositories.User{}) -// if err != nil { -// log.Fatal("cannot migrate user table") -// } -// -// return db -//} +func getOpenAIClient() openai.Client { + return openai.NewClient(option.WithAPIKey(os.Getenv("OPENAI_API_KEY"))) +} -func getEnv() { - err := godotenv.Load("./config/.env") - if err != nil { - log.Fatalf("Error getting env, not comming through %v", err) - } +func inputKnowledgeBase(chunker *chunks.Chunker, embedder *embeddings.EmbeddingService) { + + //textBytes, err := os.ReadFile("./data.md") + //if err != nil { + // log.Fatal(err) + //} + //text := string(textBytes) + // + //chunks := chunker.Chunk(text) + //fmt.Printf("Generated %d chunks\n", len(chunks)) + // + //ctx := context.Background() + //embeddings, err := embedder.Embed(ctx, chunks) + //if err != nil { + // log.Fatalf("Embedding error: %v", err) + //} + // + //// Output + //for i, emb := range embeddings { + // fmt.Printf("Chunk %d → %d-dim vector\n", i, len(emb)) + //} + // + //pineconeClient, err := pinecone.NewClient(pinecone.NewClientParams{ + // ApiKey: os.Getenv("PINECONE_API_KEY"), + //}) + //if err != nil { + // log.Fatalf("Failed to create pinecone Client: %v", err) + //} + //pineconeVectorDB := vectordb.NewPineconeVectorDB( + // os.Getenv("PINECONE_INDEX"), + // pineconeClient, + //) + // + //count, err := pineconeVectorDB.StoreEmbeddings(ctx, embeddings) + //if err != nil { + // log.Fatalf("Failed to store embeddings: %v", err) + //} + // + //fmt.Sprintf("Stored %d embeddings in Pinecone index %s\n", count, os.Getenv("PINECONE_INDEX")) } diff --git a/go.mod b/go.mod index 2c94034..0b42df0 100644 --- a/go.mod +++ b/go.mod @@ -1,12 +1,13 @@ module github.com/loukaspe/jedi-team-challenge -go 1.21 +go 1.23 toolchain go1.23.1 require ( github.com/DATA-DOG/go-sqlmock v1.5.2 github.com/golang-jwt/jwt/v4 v4.5.0 + github.com/golang/mock v1.6.0 github.com/google/uuid v1.6.0 github.com/gorilla/mux v1.8.0 github.com/joho/godotenv v1.5.1 @@ -14,23 +15,33 @@ require ( github.com/pinecone-io/go-pinecone/v3 v3.1.0 github.com/pkoukk/tiktoken-go v0.1.7 github.com/sirupsen/logrus v1.9.0 - github.com/stretchr/testify v1.8.4 + github.com/stretchr/testify v1.9.0 + github.com/swaggo/swag v1.16.4 + go.uber.org/mock v0.5.2 golang.org/x/crypto v0.32.0 gorm.io/driver/postgres v1.5.11 gorm.io/gorm v1.25.10 ) require ( + github.com/KyleBanks/depth v1.2.1 // indirect + github.com/PuerkitoBio/purell v1.1.1 // indirect + github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578 // indirect github.com/apapsch/go-jsonmerge/v2 v2.0.0 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/dlclark/regexp2 v1.10.0 // indirect + github.com/go-openapi/jsonpointer v0.19.5 // indirect + github.com/go-openapi/jsonreference v0.19.6 // indirect + github.com/go-openapi/spec v0.20.4 // indirect + github.com/go-openapi/swag v0.19.15 // indirect github.com/jackc/pgpassfile v1.0.0 // indirect github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect github.com/jackc/pgx/v5 v5.5.5 // indirect github.com/jackc/puddle/v2 v2.2.1 // indirect github.com/jinzhu/inflection v1.0.0 // indirect github.com/jinzhu/now v1.1.5 // indirect - github.com/kr/text v0.2.0 // indirect + github.com/josharian/intern v1.0.0 // indirect + github.com/mailru/easyjson v0.7.7 // indirect github.com/oapi-codegen/runtime v1.1.1 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect github.com/rogpeppe/go-internal v1.6.1 // indirect @@ -42,9 +53,11 @@ require ( golang.org/x/sync v0.11.0 // indirect golang.org/x/sys v0.30.0 // indirect golang.org/x/text v0.22.0 // indirect + golang.org/x/tools v0.22.0 // indirect google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157 // indirect google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157 // indirect google.golang.org/grpc v1.65.0 // indirect google.golang.org/protobuf v1.34.1 // indirect + gopkg.in/yaml.v2 v2.4.0 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index 11029cf..ab85064 100644 --- a/go.sum +++ b/go.sum @@ -1,5 +1,11 @@ github.com/DATA-DOG/go-sqlmock v1.5.2 h1:OcvFkGmslmlZibjAjaHm3L//6LiuBgolP7OputlJIzU= github.com/DATA-DOG/go-sqlmock v1.5.2/go.mod h1:88MAG/4G7SMwSE3CeA0ZKzrT5CiOU3OJ+JlNzwDqpNU= +github.com/KyleBanks/depth v1.2.1 h1:5h8fQADFrWtarTdtDudMmGsC7GPbOAu6RVB3ffsVFHc= +github.com/KyleBanks/depth v1.2.1/go.mod h1:jzSb9d0L43HxTQfT+oSA1EEp2q+ne2uh6XgeJcm8brE= +github.com/PuerkitoBio/purell v1.1.1 h1:WEQqlqaGbrPkxLJWfBwQmfEAE1Z7ONdDLqrN38tNFfI= +github.com/PuerkitoBio/purell v1.1.1/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0= +github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578 h1:d+Bc7a5rLufV/sSk/8dngufqelfh6jnri85riMAaF/M= +github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE= github.com/RaveNoX/go-jsoncommentstrip v1.0.0/go.mod h1:78ihd09MekBnJnxpICcwzCMzGrKSKYe4AqU6PDYYpjk= github.com/apapsch/go-jsonmerge/v2 v2.0.0 h1:axGnT1gRIfimI7gJifB699GoE/oq+F2MU7Dml6nw9rQ= github.com/apapsch/go-jsonmerge/v2 v2.0.0/go.mod h1:lvDnEdqiQrp0O42VQGgmlKpxL1AP2+08jFMw88y4klk= @@ -10,8 +16,20 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/dlclark/regexp2 v1.10.0 h1:+/GIL799phkJqYW+3YbOd8LCcbHzT0Pbo8zl70MHsq0= github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= +github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= +github.com/go-openapi/jsonpointer v0.19.5 h1:gZr+CIYByUqjcgeLXnQu2gHYQC9o73G2XUeOFYEICuY= +github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= +github.com/go-openapi/jsonreference v0.19.6 h1:UBIxjkht+AWIgYzCDSv2GN+E/togfwXUJFRTWhl2Jjs= +github.com/go-openapi/jsonreference v0.19.6/go.mod h1:diGHMEHg2IqXZGKxqyvWdfWU/aim5Dprw5bqpKkTvns= +github.com/go-openapi/spec v0.20.4 h1:O8hJrt0UMnhHcluhIdUgCLRWyM2x7QkBXRvOs7m+O1M= +github.com/go-openapi/spec v0.20.4/go.mod h1:faYFR1CvsJZ0mNsmsphTMSoRrNV3TEDoAM7FOEWeq8I= +github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= +github.com/go-openapi/swag v0.19.15 h1:D2NRCBzS9/pEY3gP9Nl8aDqGUcPFrwG2p+CNFrLyrCM= +github.com/go-openapi/swag v0.19.15/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ= github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg= github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= +github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc= +github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= @@ -32,6 +50,8 @@ github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= +github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY= +github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y= github.com/juju/gnuflag v0.0.0-20171113085948-2ce1bb71843d/go.mod h1:2PavIy+JPciBPrBUjwbNvtwB6RQlve+hkpll6QSNmOE= github.com/kisielk/sqlstruct v0.0.0-20201105191214-5f3e10d3ab46/go.mod h1:yyMNCyc/Ib3bDTKd379tNMpB/7/H5TjM2Y9QJ5THLbE= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= @@ -41,6 +61,12 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +github.com/mailru/easyjson v0.7.6/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= +github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= +github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= +github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/oapi-codegen/runtime v1.1.1 h1:EXLHh0DXIJnWhdRPN2w4MXAzFyE4CskzhNLUmtpMYro= github.com/oapi-codegen/runtime v1.1.1/go.mod h1:SK9X900oXmPWilYR5/WKPzt3Kqxn/uS/+lbpREv+eCg= github.com/openai/openai-go v1.1.0 h1:daSn+y+3QJUmLV1xfh7B8QtgJYRw1hg3yWxKtQDfROE= @@ -58,9 +84,12 @@ github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVs github.com/spkg/bom v0.0.0-20160624110644-59b7046e48ad/go.mod h1:qLr4V1qq6nMqFKkMo8ZTx3f+BZEkzsRUY10Xsm2mwU0= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk= -github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= +github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= +github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= +github.com/swaggo/swag v1.16.4 h1:clWJtd9LStiG3VeijiCfOVODP6VpHtKdQy9ELFG3s1A= +github.com/swaggo/swag v1.16.4/go.mod h1:VBsHJRsDvfYvqoiMKnsdwhNV9LEMHgEDZcyVYX0sxPg= github.com/tidwall/gjson v1.14.2/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= github.com/tidwall/gjson v1.14.4 h1:uo0p8EbA09J7RQaflQ1aBRffTR7xedD2bcIVSYxLnkM= github.com/tidwall/gjson v1.14.4/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= @@ -71,17 +100,50 @@ github.com/tidwall/pretty v1.2.1 h1:qjsOFOWWQl+N3RsoF5/ssm1pHmJJwhjlSbZ51I6wMl4= github.com/tidwall/pretty v1.2.1/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY= github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28= +github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +go.uber.org/mock v0.5.2 h1:LbtPTcP8A5k9WPXj54PPPbjcI4Y6lhyOZXn+VS7wNko= +go.uber.org/mock v0.5.2/go.mod h1:wLlUxC2vVTPTaE3UD51E0BGOAElKrILxhVSDYQLld5o= +golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.32.0 h1:euUpcYgM8WcP71gNpTqQCn6rC2t6ULUPiOzfWaXVVfc= golang.org/x/crypto v0.32.0/go.mod h1:ZnnJkOaASj8g0AjIduWNlq2NRxL0PlBrbKVyZ6V/Ugc= +golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.18.0 h1:5+9lSbEzPSdWkH32vYPBwEpX8KwDbM52Ud9xBUvNlb0= +golang.org/x/mod v0.18.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= +golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.0.0-20210421230115-4e50805a0758/go.mod h1:72T/g9IO56b78aLF+1Kcs5dz7/ng1VjMUvfKvpfy+jM= golang.org/x/net v0.34.0 h1:Mb7Mrk043xzHgnRM88suvJFwzVrRfHEHJEl5/71CKw0= golang.org/x/net v0.34.0/go.mod h1:di0qlW3YNM5oh6GqDGQr92MyTozJPmybPK4Ev/Gm31k= +golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.11.0 h1:GGz8+XQP4FvTTrjZPzNKTMFtSXH80RAzG+5ghFPgK9w= golang.org/x/sync v0.11.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= +golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210420072515-93ed5bcd2bfe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc= golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.22.0 h1:bofq7m3/HAFvbF51jz3Q9wLg3jkvSPuiZu/pD1XwgtM= golang.org/x/text v0.22.0/go.mod h1:YRoo4H8PVmsu+E3Ou7cqLVH8oXWIHVoX0jqUWALQhfY= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.22.0 h1:gqSGLZqv+AI9lIQzniJ0nZDRG5GBPsSi+DRNHWNz6yA= +golang.org/x/tools v0.22.0/go.mod h1:aCwcsjqvq7Yqt6TNyX7QMU2enbQ/Gt0bo6krSeEri+c= +golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157 h1:7whR9kGa5LUwFtpLm2ArCEejtnxlGeLbAyjFY8sGNFw= google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157/go.mod h1:99sLkeliLXfdj2J75X3Ho+rrVCaJze0uwN7zDDkjPVU= google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157 h1:Zy9XzmMEflZ/MAaA7vNcoebnRAld7FsPW1EeBB7V0m8= @@ -92,10 +154,15 @@ google.golang.org/protobuf v1.34.1 h1:9ddQBjfCyZPOHPUiPxpYESBLc+T8P3E+Vo4IbKZgFW google.golang.org/protobuf v1.34.1/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= +gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gorm.io/driver/postgres v1.5.11 h1:ubBVAfbKEUld/twyKZ0IYn9rSQh448EdelLYk9Mv314= diff --git a/internal/core/domain/chatSession.go b/internal/core/domain/chatSession.go new file mode 100644 index 0000000..5f6e88a --- /dev/null +++ b/internal/core/domain/chatSession.go @@ -0,0 +1,23 @@ +package domain + +import ( + "github.com/google/uuid" + "time" +) + +type ChatSession struct { + ID uuid.UUID + UserID uuid.UUID + Title string + CreatedAt time.Time + UpdatedAt time.Time + Messages []*Message +} + +type Message struct { + ID uuid.UUID + SessionID uuid.UUID + Sender string + Content string + Timestamp time.Time +} diff --git a/internal/core/ports/chatSessionRepositoryInterface.go b/internal/core/ports/chatSessionRepositoryInterface.go new file mode 100644 index 0000000..3fb0cfe --- /dev/null +++ b/internal/core/ports/chatSessionRepositoryInterface.go @@ -0,0 +1,14 @@ +package ports + +import ( + "context" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" +) + +type ChatSessionRepositoryInterface interface { + GetChatSession(context.Context, uuid.UUID) (*domain.ChatSession, error) + CreateChatSession(context.Context, *domain.ChatSession) (uuid.UUID, error) + UpdateChatSessionTitle(context.Context, uuid.UUID, string) error + DeleteChatSession(context.Context, uuid.UUID) error +} diff --git a/internal/core/services/chatSessionService.go b/internal/core/services/chatSessionService.go new file mode 100644 index 0000000..641c0da --- /dev/null +++ b/internal/core/services/chatSessionService.go @@ -0,0 +1,47 @@ +package services + +import ( + "context" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/ports" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" +) + +type ChatSessionServiceInterface interface { + GetChatSession(context.Context, uuid.UUID) (*domain.ChatSession, error) + CreateChatSession(context.Context, *domain.ChatSession) (uuid.UUID, error) + UpdateChatSessionTitle(context.Context, uuid.UUID, string) error + DeleteChatSession(context.Context, uuid.UUID) error +} + +type ChatSessionService struct { + logger logger.LoggerInterface + repository ports.ChatSessionRepositoryInterface +} + +func NewChatSessionService( + logger logger.LoggerInterface, + repository ports.ChatSessionRepositoryInterface, +) *ChatSessionService { + return &ChatSessionService{ + logger: logger, + repository: repository, + } +} + +func (s ChatSessionService) CreateChatSession(ctx context.Context, session *domain.ChatSession) (uuid.UUID, error) { + return s.repository.CreateChatSession(ctx, session) +} + +func (s ChatSessionService) UpdateChatSessionTitle(ctx context.Context, uuid uuid.UUID, title string) error { + return s.repository.UpdateChatSessionTitle(ctx, uuid, title) +} + +func (s ChatSessionService) DeleteChatSession(ctx context.Context, uuid uuid.UUID) error { + return s.repository.DeleteChatSession(ctx, uuid) +} + +func (s ChatSessionService) GetChatSession(ctx context.Context, uuid uuid.UUID) (*domain.ChatSession, error) { + return s.repository.GetChatSession(ctx, uuid) +} diff --git a/internal/handlers/chatSessions/createHandler.go b/internal/handlers/chatSessions/createHandler.go new file mode 100644 index 0000000..4f8bb89 --- /dev/null +++ b/internal/handlers/chatSessions/createHandler.go @@ -0,0 +1,133 @@ +package chatSessions + +import ( + "encoding/json" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" +) + +type CreateUserChatSessionHandler struct { + ChatSessionService services.ChatSessionServiceInterface + logger logger.LoggerInterface +} + +func NewCreateUserChatSessionHandler( + service services.ChatSessionServiceInterface, + logger logger.LoggerInterface, +) *CreateUserChatSessionHandler { + return &CreateUserChatSessionHandler{ + ChatSessionService: service, + logger: logger, + } +} + +// @Summary Creates chat session +// @Description Creates a chat session for User +// @Security BearerAuth +// @Param user_id path int true "user id" +// @Success 201 {object} ChatSessionResponse +// @Failure 400 {object} ChatSessionResponse "Error in message payload" +// @Failure 401 {object} ChatSessionResponse "Authentication error" +// @Failure 500 {object} ChatSessionResponse "Internal Server Error" +// @Router /users/user_id/chat-sessions [post] +func (handler *CreateUserChatSessionHandler) CreateUserChatSessionAssetController(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + + ctx := r.Context() + + var err error + response := &ChatSessionResponse{} + + userIdAsString := mux.Vars(r)["user_id"] + if userIdAsString == "" { + response.ErrorMessage = "missing user id" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + userId, err := uuid.Parse(userIdAsString) + if err != nil { + handler.logger.Error("Error in creating chat session", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed user uuid" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + //err = json.NewDecoder(r.Body).Decode(chatSessionRequest) + //if err != nil { + // handler.logger.Error("Error in creating chat session", + // map[string]interface{}{ + // "errorMessage": err.Error(), + // }) + // + // response.ErrorMessage = "malformed create chat session request" + // + // handler.JsonResponse(w, http.StatusBadRequest, response) + // + // return + //} + + insertedUUID, err := handler.ChatSessionService.CreateChatSession( + ctx, + &domain.ChatSession{ + UserID: userId, + }, + ) + if userNotFoundError, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Error in creating chat session", + map[string]interface{}{ + "errorMessage": userNotFoundError.Unwrap(), + }) + + response.ErrorMessage = err.Error() + handler.JsonResponse(w, http.StatusNotFound, response) + + return + } + + if err != nil { + handler.logger.Error("Error in creating chat session", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "error in creating chat session" + handler.JsonResponse(w, http.StatusInternalServerError, response) + + return + } + + response.ID = insertedUUID.String() + handler.JsonResponse(w, http.StatusCreated, response) +} + +func (handler *CreateUserChatSessionHandler) JsonResponse( + w http.ResponseWriter, + statusCode int, + response *ChatSessionResponse, +) { + w.WriteHeader(statusCode) + err := json.NewEncoder(w).Encode(response) + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + response.ErrorMessage = "error in creating chat session - json response" + + handler.logger.Error("Error in creating chat session - json response", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + } +} diff --git a/internal/handlers/chatSessions/createHandler_test.go b/internal/handlers/chatSessions/createHandler_test.go new file mode 100644 index 0000000..2e2c738 --- /dev/null +++ b/internal/handlers/chatSessions/createHandler_test.go @@ -0,0 +1,258 @@ +package chatSessions + +import ( + "context" + "encoding/json" + "errors" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + mock_services "github.com/loukaspe/jedi-team-challenge/mocks/mock_internal/core/services" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/stretchr/testify/assert" + "go.uber.org/mock/gomock" + "io" + "net/http/httptest" + "testing" +) + +func TestCreateUserChatSessionHandler_CreateUserChatSessionController(t *testing.T) { + logger := logger.NewLogger(context.Background()) + mockCtrl := gomock.NewController(t) + defer mockCtrl.Finish() + + mockService := mock_services.NewMockChatSessionServiceInterface(mockCtrl) + + type args struct { + userId uuid.UUID + } + + tests := []struct { + name string + args args + mockServiceInsertedID uuid.UUID + expected []byte + expectedStatusCode int + }{ + { + name: "valid", + args: args{ + userId: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + mockServiceInsertedID: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + expected: json.RawMessage(`{"id":"22345688-0000-0000-0000-000000000000"} +`), + expectedStatusCode: 201, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + mockRequest := httptest.NewRequest( + "POST", + "/users/"+tt.args.userId.String()+"/chat-sessions", + nil, + ) + + vars := map[string]string{ + "user_id": tt.args.userId.String(), + } + mockRequest = mux.SetURLVars(mockRequest, vars) + + mockRequest.Header.Set("Content-Type", "application/json") + mockResponseRecorder := httptest.NewRecorder() + + mockService.EXPECT().CreateChatSession( + gomock.Any(), + &domain.ChatSession{UserID: tt.args.userId}, + ).Return(tt.mockServiceInsertedID, nil) + + handler := &CreateUserChatSessionHandler{ + ChatSessionService: mockService, + logger: logger, + } + sut := handler.CreateUserChatSessionAssetController + + sut(mockResponseRecorder, mockRequest) + + mockResponse := mockResponseRecorder.Result() + actual, err := io.ReadAll(mockResponse.Body) + if err != nil { + t.Errorf("error with response reading: %v", err) + return + } + actualStatusCode := mockResponse.StatusCode + + assert.Equal(t, string(tt.expected), string(actual)) + assert.Equal(t, tt.expectedStatusCode, actualStatusCode) + }) + } +} + +func TestCreateUserChatSessionHandler_CreateUserChatSessionAssetControllerHasBadRequestError(t *testing.T) { + logger := logger.NewLogger(context.Background()) + mockCtrl := gomock.NewController(t) + defer mockCtrl.Finish() + + mockService := mock_services.NewMockChatSessionServiceInterface(mockCtrl) + + type args struct { + userId uuid.UUID + userIdAsString string + } + + tests := []struct { + name string + args args + mockServiceInsertedID uuid.UUID + expected []byte + expectedStatusCode int + }{ + { + name: "empty user id", + args: args{ + userIdAsString: "", + }, + + expected: json.RawMessage(`{"errorMessage":"missing user id"} +`), + expectedStatusCode: 400, + }, + { + name: "user id not a uuid", + args: args{ + userIdAsString: "55", + }, + expected: json.RawMessage(`{"errorMessage":"malformed user uuid"} +`), + expectedStatusCode: 400, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + + mockRequest := httptest.NewRequest( + "POST", + "/users/"+tt.args.userIdAsString+"/chat-sessions", + nil, + ) + + vars := map[string]string{ + "user_id": tt.args.userIdAsString, + } + mockRequest = mux.SetURLVars(mockRequest, vars) + + mockRequest.Header.Set("Content-Type", "application/json") + mockResponseRecorder := httptest.NewRecorder() + + handler := &CreateUserChatSessionHandler{ + ChatSessionService: mockService, + logger: logger, + } + sut := handler.CreateUserChatSessionAssetController + + sut(mockResponseRecorder, mockRequest) + + mockResponse := mockResponseRecorder.Result() + actual, err := io.ReadAll(mockResponse.Body) + if err != nil { + t.Errorf("error with response reading: %v", err) + return + } + actualStatusCode := mockResponse.StatusCode + + assert.Equal(t, string(tt.expected), string(actual)) + assert.Equal(t, tt.expectedStatusCode, actualStatusCode) + }) + } +} + +func TestCreateUserChatSessionHandler_CreateUserChatSessionAssetControllerHasServiceError(t *testing.T) { + logger := logger.NewLogger(context.Background()) + mockCtrl := gomock.NewController(t) + defer mockCtrl.Finish() + + mockService := mock_services.NewMockChatSessionServiceInterface(mockCtrl) + + type args struct { + userId uuid.UUID + } + + tests := []struct { + name string + args args + mockServiceInsertedID uuid.UUID + mockServiceResponseError error + expected []byte + expectedStatusCode int + }{ + { + name: "service random error", + args: args{ + userId: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + mockServiceInsertedID: uuid.UUID{}, + mockServiceResponseError: errors.New("random error"), + expected: json.RawMessage(`{"errorMessage":"error in creating chat session"} +`), + expectedStatusCode: 500, + }, + { + name: "service user not found error", + args: args{ + userId: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + mockServiceInsertedID: uuid.UUID{}, + mockServiceResponseError: apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("user id 667 not found"), + }, + expected: json.RawMessage(`{} +`), + expectedStatusCode: 404, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + mockRequest := httptest.NewRequest( + "POST", + "/users/"+tt.args.userId.String()+"/chat-sessions", + nil, + ) + + vars := map[string]string{ + "user_id": tt.args.userId.String(), + } + mockRequest = mux.SetURLVars(mockRequest, vars) + + mockRequest.Header.Set("Content-Type", "application/json") + mockResponseRecorder := httptest.NewRecorder() + + mockService.EXPECT().CreateChatSession( + gomock.Any(), + &domain.ChatSession{UserID: tt.args.userId}, + ).Return(tt.mockServiceInsertedID, tt.mockServiceResponseError) + + handler := &CreateUserChatSessionHandler{ + ChatSessionService: mockService, + logger: logger, + } + sut := handler.CreateUserChatSessionAssetController + + sut(mockResponseRecorder, mockRequest) + + mockResponse := mockResponseRecorder.Result() + actual, err := io.ReadAll(mockResponse.Body) + if err != nil { + t.Errorf("error with response reading: %v", err) + return + } + actualStatusCode := mockResponse.StatusCode + + assert.Equal(t, string(tt.expected), string(actual)) + assert.Equal(t, tt.expectedStatusCode, actualStatusCode) + }) + } +} diff --git a/internal/handlers/chatSessions/dto.go b/internal/handlers/chatSessions/dto.go new file mode 100644 index 0000000..d51d030 --- /dev/null +++ b/internal/handlers/chatSessions/dto.go @@ -0,0 +1,34 @@ +package chatSessions + +type ChatSessionResponse struct { + ID string `json:"id,omitempty"` + Title string `json:"title,omitempty"` + CreatedAt string `json:"createdAt,omitempty"` + UpdatedAt string `json:"updatedAt,omitempty"` + Messages []Message `json:"messages,omitempty"` + ErrorMessage string `json:"errorMessage,omitempty"` +} + +type Message struct { + ID string `json:"id"` + Sender string `json:"sender" enum:"USER,SYSTEM" example:"USER"` + Content string `json:"content"` + Timestamp string `json:"timestamp"` +} + +//type CreateChatSessionRequest struct { +// UserID string `json:"userId"` +//} + +type SendMessageRequest struct { + UserID string `json:"userId"` + SessionID string `json:"sessionId"` + Content string `json:"content"` +} + +type SendMessageResponse struct { + Body struct { + UserMessage Message `json:"userMessage"` + SystemMessage Message `json:"systemMessage"` + } +} diff --git a/internal/repositories/chatSessionDao.go b/internal/repositories/chatSessionDao.go new file mode 100644 index 0000000..8ab3531 --- /dev/null +++ b/internal/repositories/chatSessionDao.go @@ -0,0 +1,23 @@ +package repositories + +import ( + "github.com/google/uuid" + "time" +) + +type ChatSession struct { + ID uuid.UUID `gorm:"type:uuid;primaryKey;default:uuid_generate_v4()"` + UserID uuid.UUID `gorm:"type:uuid;not null;index"` + Title string `gorm:"type:text"` + CreatedAt time.Time `gorm:"autoCreateTime"` + UpdatedAt time.Time `gorm:"autoUpdateTime"` + Messages []Message `gorm:"foreignKey:SessionID;constraint:OnDelete:CASCADE"` +} + +type Message struct { + ID uuid.UUID `gorm:"type:uuid;primaryKey;default:uuid_generate_v4()"` + SessionID uuid.UUID `gorm:"type:uuid;not null;index"` + Sender string `gorm:"type:text;not null"` + Content string `gorm:"type:text;not null"` + Timestamp time.Time `gorm:"not null"` +} diff --git a/internal/repositories/chatSessionRepositoryCreate.go b/internal/repositories/chatSessionRepositoryCreate.go new file mode 100644 index 0000000..40c450d --- /dev/null +++ b/internal/repositories/chatSessionRepositoryCreate.go @@ -0,0 +1,36 @@ +package repositories + +import ( + "context" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "gorm.io/gorm" +) + +type ChatSessionRepository struct { + db *gorm.DB +} + +func NewChatSessionRepository(db *gorm.DB) *ChatSessionRepository { + return &ChatSessionRepository{db: db} +} + +func (repo *ChatSessionRepository) CreateChatSession( + ctx context.Context, + chat *domain.ChatSession, +) (uuid.UUID, error) { + var err error + + modelChat := ChatSession{ + UserID: chat.UserID, + Title: chat.Title, + Messages: nil, + } + + err = repo.db.WithContext(ctx).Create(&modelChat).Error + if err != nil { + return uuid.UUID{}, err + } + + return modelChat.ID, nil +} diff --git a/internal/repositories/chatSessionRepositoryCreate_test.go b/internal/repositories/chatSessionRepositoryCreate_test.go new file mode 100644 index 0000000..a11cdb1 --- /dev/null +++ b/internal/repositories/chatSessionRepositoryCreate_test.go @@ -0,0 +1,136 @@ +package repositories + +import ( + "context" + "errors" + "github.com/DATA-DOG/go-sqlmock" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/stretchr/testify/assert" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "regexp" + "testing" +) + +func TestChatRepository_CreateChatSession(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + chat *domain.ChatSession + } + tests := []struct { + name string + args args + mockSqlChatQueryExpected string + mockInsertedChatIdReturned uuid.UUID + expectedChatUid uuid.UUID + }{ + { + name: "valid", + args: args{ + chat: &domain.ChatSession{ + UserID: uuid.UUID{ + 0x12, 0x34, 0x56, 0x78, + }, + Title: "mockTitle", + Messages: nil, + }, + }, + mockSqlChatQueryExpected: `INSERT INTO "chat_sessions" ("user_id","title","created_at","updated_at") VALUES ($1,$2,$3,$4) RETURNING "id"`, + mockInsertedChatIdReturned: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + expectedChatUid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &ChatSessionRepository{ + db: gormDb, + } + + mockDb.ExpectBegin() + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlChatQueryExpected)). + WithArgs( + tt.args.chat.UserID, tt.args.chat.Title, sqlmock.AnyArg(), sqlmock.AnyArg(), + ). + WillReturnRows(sqlmock.NewRows([]string{"id"}).AddRow(tt.mockInsertedChatIdReturned)) + mockDb.ExpectCommit() + + actual, err := repo.CreateChatSession(context.Background(), tt.args.chat) + if err != nil { + t.Errorf("CreateChat() error = %v", err) + } + + assert.Equal(t, tt.expectedChatUid, actual) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} + +func TestChatRepository_CreateChatSessionWithError(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + chat *domain.ChatSession + } + tests := []struct { + name string + args args + mockSqlChatQueryExpected string + expectedErrorMessage string + }{ + { + name: "random error", + args: args{ + chat: &domain.ChatSession{ + Title: "mockTitle", + UserID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + }, + mockSqlChatQueryExpected: `INSERT INTO "chat_sessions" ("user_id","title","created_at","updated_at") VALUES ($1,$2,$3,$4) RETURNING "id"`, + expectedErrorMessage: "random error", + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &ChatSessionRepository{ + db: gormDb, + } + + mockDb.ExpectBegin() + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlChatQueryExpected)). + WithArgs( + tt.args.chat.UserID, tt.args.chat.Title, sqlmock.AnyArg(), sqlmock.AnyArg(), + ). + WillReturnError(errors.New(tt.expectedErrorMessage)) + mockDb.ExpectRollback() + + _, err := repo.CreateChatSession(context.Background(), tt.args.chat) + actualErrorMessage := err.Error() + + assert.Equal(t, tt.expectedErrorMessage, actualErrorMessage) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} diff --git a/internal/repositories/userDao.go b/internal/repositories/userDao.go index 3a6118b..51cfe6a 100644 --- a/internal/repositories/userDao.go +++ b/internal/repositories/userDao.go @@ -1,9 +1,15 @@ package repositories -import "gorm.io/gorm" +import ( + "github.com/google/uuid" + "time" +) type User struct { - gorm.Model - Username string `gorm:"not null;"` - Password string `gorm:"not null;"` + ID uuid.UUID `gorm:"type:uuid;primaryKey;default:uuid_generate_v4()"` + CreatedAt time.Time `gorm:"autoCreateTime"` + UpdatedAt time.Time `gorm:"autoUpdateTime"` + Username string `gorm:"not null;"` + Password string `gorm:"not null;"` + ChatSessions []ChatSession } diff --git a/mocks/mock_internal/core/services/chatSessionService.go b/mocks/mock_internal/core/services/chatSessionService.go new file mode 100644 index 0000000..3a55e04 --- /dev/null +++ b/mocks/mock_internal/core/services/chatSessionService.go @@ -0,0 +1,100 @@ +// Code generated by MockGen. DO NOT EDIT. +// Source: ../internal/core/services/chatSessionService.go +// +// Generated by this command: +// +// mockgen -source=../internal/core/services/chatSessionService.go -destination=../mocks/mock_internal/core/services/chatSessionService.go +// + +// Package mock_services is a generated GoMock package. +package mock_services + +import ( + context "context" + reflect "reflect" + + uuid "github.com/google/uuid" + domain "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + gomock "go.uber.org/mock/gomock" +) + +// MockChatSessionServiceInterface is a mock of ChatSessionServiceInterface interface. +type MockChatSessionServiceInterface struct { + ctrl *gomock.Controller + recorder *MockChatSessionServiceInterfaceMockRecorder +} + +// MockChatSessionServiceInterfaceMockRecorder is the mock recorder for MockChatSessionServiceInterface. +type MockChatSessionServiceInterfaceMockRecorder struct { + mock *MockChatSessionServiceInterface +} + +// NewMockChatSessionServiceInterface creates a new mock instance. +func NewMockChatSessionServiceInterface(ctrl *gomock.Controller) *MockChatSessionServiceInterface { + mock := &MockChatSessionServiceInterface{ctrl: ctrl} + mock.recorder = &MockChatSessionServiceInterfaceMockRecorder{mock} + return mock +} + +// EXPECT returns an object that allows the caller to indicate expected use. +func (m *MockChatSessionServiceInterface) EXPECT() *MockChatSessionServiceInterfaceMockRecorder { + return m.recorder +} + +// CreateChatSession mocks base method. +func (m *MockChatSessionServiceInterface) CreateChatSession(arg0 context.Context, arg1 *domain.ChatSession) (uuid.UUID, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "CreateChatSession", arg0, arg1) + ret0, _ := ret[0].(uuid.UUID) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// CreateChatSession indicates an expected call of CreateChatSession. +func (mr *MockChatSessionServiceInterfaceMockRecorder) CreateChatSession(arg0, arg1 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "CreateChatSession", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).CreateChatSession), arg0, arg1) +} + +// DeleteChatSession mocks base method. +func (m *MockChatSessionServiceInterface) DeleteChatSession(arg0 context.Context, arg1 uuid.UUID) error { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "DeleteChatSession", arg0, arg1) + ret0, _ := ret[0].(error) + return ret0 +} + +// DeleteChatSession indicates an expected call of DeleteChatSession. +func (mr *MockChatSessionServiceInterfaceMockRecorder) DeleteChatSession(arg0, arg1 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "DeleteChatSession", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).DeleteChatSession), arg0, arg1) +} + +// GetChatSession mocks base method. +func (m *MockChatSessionServiceInterface) GetChatSession(arg0 context.Context, arg1 uuid.UUID) (*domain.ChatSession, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "GetChatSession", arg0, arg1) + ret0, _ := ret[0].(*domain.ChatSession) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// GetChatSession indicates an expected call of GetChatSession. +func (mr *MockChatSessionServiceInterfaceMockRecorder) GetChatSession(arg0, arg1 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetChatSession", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).GetChatSession), arg0, arg1) +} + +// UpdateChatSessionTitle mocks base method. +func (m *MockChatSessionServiceInterface) UpdateChatSessionTitle(arg0 context.Context, arg1 uuid.UUID, arg2 string) error { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "UpdateChatSessionTitle", arg0, arg1, arg2) + ret0, _ := ret[0].(error) + return ret0 +} + +// UpdateChatSessionTitle indicates an expected call of UpdateChatSessionTitle. +func (mr *MockChatSessionServiceInterfaceMockRecorder) UpdateChatSessionTitle(arg0, arg1, arg2 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "UpdateChatSessionTitle", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).UpdateChatSessionTitle), arg0, arg1, arg2) +} diff --git a/pkg/chunks/chunker.go b/pkg/chunks/chunker.go index 873a8ab..5f95409 100644 --- a/pkg/chunks/chunker.go +++ b/pkg/chunks/chunker.go @@ -5,18 +5,18 @@ import ( "strings" ) -type TiktokenChunker struct { +type Chunker struct { Encoder *tiktoken.Tiktoken MaxTokensPerChunk int } // TiktokenEncoding string // enc, err := tiktoken.GetEncoding(encName) -func NewTiktokenChunker(encoder *tiktoken.Tiktoken, maxTokens int) (*TiktokenChunker, error) { - return &TiktokenChunker{Encoder: encoder, MaxTokensPerChunk: maxTokens}, nil +func NewChunker(encoder *tiktoken.Tiktoken, maxTokens int) (*Chunker, error) { + return &Chunker{Encoder: encoder, MaxTokensPerChunk: maxTokens}, nil } -func (c *TiktokenChunker) Chunk(text string) []string { +func (c *Chunker) Chunk(text string) []string { var chunks []string sentences := splitToSentences(text) diff --git a/pkg/errors/errors.go b/pkg/errors/errors.go index 2cf0876..b90e9a7 100644 --- a/pkg/errors/errors.go +++ b/pkg/errors/errors.go @@ -1,61 +1,15 @@ package apierrors -type UserNotFoundErrorWrapper struct { - ReturnedStatusCode int - OriginalError error +type ResourceNotFoundErrorWrapper struct { + OriginalError error } // Error the original error message remains as it is for logging reasons etc. // and the wrapper error message is empty because we don't want the client to see anything -func (err UserNotFoundErrorWrapper) Error() string { +func (err ResourceNotFoundErrorWrapper) Error() string { return "" } -func (err UserNotFoundErrorWrapper) Unwrap() error { - return err.OriginalError -} - -type AssetNotFoundErrorWrapper struct { - ReturnedStatusCode int - OriginalError error -} - -// Error the original error message remains as it is for logging reasons etc. -// and the wrapper error message is empty because we don't want the client to see anything -func (err AssetNotFoundErrorWrapper) Error() string { - return "" -} - -func (err AssetNotFoundErrorWrapper) Unwrap() error { - return err.OriginalError -} - -type UnknownAssetTypeErrorWrapper struct { - ReturnedStatusCode int - OriginalError error -} - -// Error the original error message remains as it is for logging reasons etc. -// and the wrapper error message is empty because we don't want the client to see anything -func (err UnknownAssetTypeErrorWrapper) Error() string { - return "" -} - -func (err UnknownAssetTypeErrorWrapper) Unwrap() error { - return err.OriginalError -} - -type NoFavouriteAssetsErrorWrapper struct { - ReturnedStatusCode int - OriginalError error -} - -// Error the original error message remains as it is for logging reasons etc. -// and the wrapper error message is empty because we don't want the client to see anything -func (err NoFavouriteAssetsErrorWrapper) Error() string { - return "" -} - -func (err NoFavouriteAssetsErrorWrapper) Unwrap() error { +func (err ResourceNotFoundErrorWrapper) Unwrap() error { return err.OriginalError } diff --git a/pkg/server/routes.go b/pkg/server/routes.go index fc7ba53..641303a 100644 --- a/pkg/server/routes.go +++ b/pkg/server/routes.go @@ -3,15 +3,34 @@ package server import ( "github.com/loukaspe/jedi-team-challenge/internal/core/services" "github.com/loukaspe/jedi-team-challenge/internal/handlers" + "github.com/loukaspe/jedi-team-challenge/internal/handlers/chatSessions" + "github.com/loukaspe/jedi-team-challenge/internal/repositories" "github.com/loukaspe/jedi-team-challenge/pkg/auth" "net/http" "os" ) +// @title Louk Chatwalker +// @version 1.0 +// @description GWI's Jedi Team Challenge + +// @host localhost:8080 +// @BasePath / + +// @contact.name Loukas Peteinaris +// @contact.url loukas.peteinaris@gmail.com + +// @securityDefinitions.apikey BearerAuth +// @in header +// @name Authorization +// @description Header value should be in the form of `Bearer ` + +// @accept json +// @produce json func (s *Server) initializeRoutes() { // health check - healthCheckHandler := handlers.NewHealthCheckHandler() + healthCheckHandler := handlers.NewHealthCheckHandler(s.DB) s.router.HandleFunc("/health-check", healthCheckHandler.HealthCheckController).Methods("GET") // auth @@ -28,6 +47,11 @@ func (s *Server) initializeRoutes() { protected := s.router.PathPrefix("/").Subrouter() protected.Use(jwtMiddleware.AuthenticationMW) - //protected.HandleFunc("/users/{user_id:[0-9]+}/favourites", createUserFavouriteHandler.AddUserFavouriteAssetController).Methods("POST") + chatSessionRepository := repositories.NewChatSessionRepository(s.DB) + chatSessionService := services.NewChatSessionService(s.logger, chatSessionRepository) + + createChatSessionHandler := chatSessions.NewCreateUserChatSessionHandler(chatSessionService, s.logger) + + protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionAssetController).Methods("POST") } From b07c79e09ebdaece5ece1ff62a887b30c44eea08 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 00:13:19 +0300 Subject: [PATCH 07/52] Adds get chat sessions --- cmd/http/main.go | 8 +- internal/core/domain/chatSession.go | 10 +- .../ports/chatSessionRepositoryInterface.go | 1 + internal/core/services/chatSessionService.go | 5 + internal/handlers/chatSessions/dto.go | 44 +- internal/handlers/chatSessions/getHandler.go | 182 +++++++++ .../handlers/chatSessions/getHandler_test.go | 174 ++++++++ internal/repositories/chatSessionDao.go | 12 +- .../repositories/chatSessionRepositoryGet.go | 107 +++++ .../chatSessionRepositoryGet_test.go | 378 ++++++++++++++++++ .../core/services/chatSessionService.go | 15 + pkg/server/routes.go | 4 + 12 files changed, 926 insertions(+), 14 deletions(-) create mode 100644 internal/handlers/chatSessions/getHandler.go create mode 100644 internal/handlers/chatSessions/getHandler_test.go create mode 100644 internal/repositories/chatSessionRepositoryGet.go create mode 100644 internal/repositories/chatSessionRepositoryGet_test.go diff --git a/cmd/http/main.go b/cmd/http/main.go index 0a50736..902330a 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -67,6 +67,7 @@ func getDB() *gorm.DB { // Drops added in order to start with clean DB on App start for // assessment reasons db.Migrator().DropTable("users") + db.Migrator().DropTable("messages") db.Migrator().DropTable("chat_sessions") err = db.AutoMigrate(&repositories.User{}) @@ -76,7 +77,7 @@ func getDB() *gorm.DB { // TODO: remove admin := repositories.User{ - ID: uuid.New(), + ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, Username: "loukas", Password: "loukastest", } @@ -91,6 +92,11 @@ func getDB() *gorm.DB { log.Fatal("cannot migrate chat sessions table") } + err = db.AutoMigrate(&repositories.Message{}) + if err != nil { + log.Fatal("cannot migrate messages table") + } + return db } diff --git a/internal/core/domain/chatSession.go b/internal/core/domain/chatSession.go index 5f6e88a..dcb3b88 100644 --- a/internal/core/domain/chatSession.go +++ b/internal/core/domain/chatSession.go @@ -15,9 +15,9 @@ type ChatSession struct { } type Message struct { - ID uuid.UUID - SessionID uuid.UUID - Sender string - Content string - Timestamp time.Time + ID uuid.UUID + ChatSessionID uuid.UUID + Sender string + Content string + Timestamp time.Time } diff --git a/internal/core/ports/chatSessionRepositoryInterface.go b/internal/core/ports/chatSessionRepositoryInterface.go index 3fb0cfe..6e92c3e 100644 --- a/internal/core/ports/chatSessionRepositoryInterface.go +++ b/internal/core/ports/chatSessionRepositoryInterface.go @@ -8,6 +8,7 @@ import ( type ChatSessionRepositoryInterface interface { GetChatSession(context.Context, uuid.UUID) (*domain.ChatSession, error) + GetUserChatSessions(context.Context, uuid.UUID) ([]*domain.ChatSession, error) CreateChatSession(context.Context, *domain.ChatSession) (uuid.UUID, error) UpdateChatSessionTitle(context.Context, uuid.UUID, string) error DeleteChatSession(context.Context, uuid.UUID) error diff --git a/internal/core/services/chatSessionService.go b/internal/core/services/chatSessionService.go index 641c0da..28b5953 100644 --- a/internal/core/services/chatSessionService.go +++ b/internal/core/services/chatSessionService.go @@ -10,6 +10,7 @@ import ( type ChatSessionServiceInterface interface { GetChatSession(context.Context, uuid.UUID) (*domain.ChatSession, error) + GetUserChatSessions(context.Context, uuid.UUID) ([]*domain.ChatSession, error) CreateChatSession(context.Context, *domain.ChatSession) (uuid.UUID, error) UpdateChatSessionTitle(context.Context, uuid.UUID, string) error DeleteChatSession(context.Context, uuid.UUID) error @@ -45,3 +46,7 @@ func (s ChatSessionService) DeleteChatSession(ctx context.Context, uuid uuid.UUI func (s ChatSessionService) GetChatSession(ctx context.Context, uuid uuid.UUID) (*domain.ChatSession, error) { return s.repository.GetChatSession(ctx, uuid) } + +func (s ChatSessionService) GetUserChatSessions(ctx context.Context, uuid uuid.UUID) ([]*domain.ChatSession, error) { + return s.repository.GetUserChatSessions(ctx, uuid) +} diff --git a/internal/handlers/chatSessions/dto.go b/internal/handlers/chatSessions/dto.go index d51d030..8fd2759 100644 --- a/internal/handlers/chatSessions/dto.go +++ b/internal/handlers/chatSessions/dto.go @@ -1,5 +1,14 @@ package chatSessions +import ( + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" +) + +type UserChatSessionsResponse struct { + Sessions []ChatSessionResponse `json:"sessions,omitempty"` + ErrorMessage string `json:"errorMessage,omitempty"` +} + type ChatSessionResponse struct { ID string `json:"id,omitempty"` Title string `json:"title,omitempty"` @@ -9,6 +18,37 @@ type ChatSessionResponse struct { ErrorMessage string `json:"errorMessage,omitempty"` } +func ChatSessionResponseFromModel(domainChatSession *domain.ChatSession) *ChatSessionResponse { + messages := make([]Message, len(domainChatSession.Messages)) + for i, msg := range domainChatSession.Messages { + messages[i] = Message{ + ID: msg.ID.String(), + Sender: msg.Sender, + Content: msg.Content, + Timestamp: msg.Timestamp.String(), + } + } + + return &ChatSessionResponse{ + ID: domainChatSession.ID.String(), + Title: domainChatSession.Title, + CreatedAt: domainChatSession.CreatedAt.String(), + UpdatedAt: domainChatSession.UpdatedAt.String(), + Messages: messages, + } +} + +func UserChatSessionsResponseFromModel(domainChatSession []*domain.ChatSession) *UserChatSessionsResponse { + sessions := make([]ChatSessionResponse, len(domainChatSession)) + for i, session := range domainChatSession { + sessions[i] = *ChatSessionResponseFromModel(session) + } + + return &UserChatSessionsResponse{ + Sessions: sessions, + } +} + type Message struct { ID string `json:"id"` Sender string `json:"sender" enum:"USER,SYSTEM" example:"USER"` @@ -17,11 +57,11 @@ type Message struct { } //type CreateChatSessionRequest struct { -// UserID string `json:"userId"` +// UserID string `json:"chatSessionID"` //} type SendMessageRequest struct { - UserID string `json:"userId"` + UserID string `json:"chatSessionID"` SessionID string `json:"sessionId"` Content string `json:"content"` } diff --git a/internal/handlers/chatSessions/getHandler.go b/internal/handlers/chatSessions/getHandler.go new file mode 100644 index 0000000..aa053a8 --- /dev/null +++ b/internal/handlers/chatSessions/getHandler.go @@ -0,0 +1,182 @@ +package chatSessions + +import ( + "encoding/json" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" +) + +type GetChatSessionHandler struct { + ChatSessionService services.ChatSessionServiceInterface + logger logger.LoggerInterface +} + +func NewGetChatSessionHandler( + service services.ChatSessionServiceInterface, + logger logger.LoggerInterface, +) *GetChatSessionHandler { + return &GetChatSessionHandler{ + ChatSessionService: service, + logger: logger, + } +} + +func (handler *GetChatSessionHandler) GetUserChatSessionsController(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + + var err error + response := &UserChatSessionsResponse{} + + ctx := r.Context() + + userIdAsString := mux.Vars(r)["user_id"] + if userIdAsString == "" { + response.ErrorMessage = "missing user id" + + handler.JsonUserChatSessionResponse(w, http.StatusBadRequest, response) + + return + } + + userId, err := uuid.Parse(userIdAsString) + if err != nil { + handler.logger.Error("Error in getting chat session", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed user uuid" + + handler.JsonUserChatSessionResponse(w, http.StatusBadRequest, response) + + return + } + + usersChatSessions, err := handler.ChatSessionService.GetUserChatSessions(ctx, userId) + if userNotFoundError, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Error in getting users chat sessions", + map[string]interface{}{ + "errorMessage": userNotFoundError.Unwrap(), + }) + + response.ErrorMessage = err.Error() + handler.JsonUserChatSessionResponse(w, http.StatusNotFound, response) + + return + } + + if err != nil { + handler.logger.Error("Error in getting user's favourite assets", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "error in getting user favourite assets" + handler.JsonUserChatSessionResponse(w, http.StatusInternalServerError, response) + + return + } + + response = UserChatSessionsResponseFromModel(usersChatSessions) + handler.JsonUserChatSessionResponse(w, http.StatusOK, response) +} + +func (handler *GetChatSessionHandler) GetChatSessionController(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + + var err error + response := &ChatSessionResponse{} + + ctx := r.Context() + + sessionIDAsString := mux.Vars(r)["session_id"] + if sessionIDAsString == "" { + response.ErrorMessage = "missing session id" + + handler.JsonChatSessionResponse(w, http.StatusBadRequest, response) + + return + } + + sessionID, err := uuid.Parse(sessionIDAsString) + if err != nil { + handler.logger.Error("Error in getting chat session", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed session uuid" + + handler.JsonChatSessionResponse(w, http.StatusBadRequest, response) + + return + } + + chatSession, err := handler.ChatSessionService.GetChatSession(ctx, sessionID) + if chatSessionNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Error in getting chat session", + map[string]interface{}{ + "errorMessage": chatSessionNotFound.Unwrap(), + }) + + response.ErrorMessage = err.Error() + handler.JsonChatSessionResponse(w, http.StatusNotFound, response) + + return + } + + if err != nil { + handler.logger.Error("Error in getting chat session", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "error in getting chat session" + handler.JsonChatSessionResponse(w, http.StatusInternalServerError, response) + + return + } + + response = ChatSessionResponseFromModel(chatSession) + handler.JsonChatSessionResponse(w, http.StatusOK, response) +} + +func (handler *GetChatSessionHandler) JsonChatSessionResponse( + w http.ResponseWriter, + statusCode int, + response *ChatSessionResponse, +) { + w.WriteHeader(statusCode) + err := json.NewEncoder(w).Encode(response) + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + response.ErrorMessage = "error in getting chat session - json response" + + handler.logger.Error("Error in getting chat session - json response", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + } +} + +func (handler *GetChatSessionHandler) JsonUserChatSessionResponse( + w http.ResponseWriter, + statusCode int, + response *UserChatSessionsResponse, +) { + w.WriteHeader(statusCode) + err := json.NewEncoder(w).Encode(response) + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + response.ErrorMessage = "error in getting user chat sessions - json response" + + handler.logger.Error("Error in getting user chat sessions - json response", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + } +} diff --git a/internal/handlers/chatSessions/getHandler_test.go b/internal/handlers/chatSessions/getHandler_test.go new file mode 100644 index 0000000..0bacf5f --- /dev/null +++ b/internal/handlers/chatSessions/getHandler_test.go @@ -0,0 +1,174 @@ +package chatSessions + +import ( + "context" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + mock_services "github.com/loukaspe/jedi-team-challenge/mocks/mock_internal/core/services" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/stretchr/testify/assert" + "go.uber.org/mock/gomock" + "io" + "net/http/httptest" + "testing" +) + +func TestGetChatSessionHandler_GetChatSessionController(t *testing.T) { + logger := logger.NewLogger(context.Background()) + mockCtrl := gomock.NewController(t) + defer mockCtrl.Finish() + + mockService := mock_services.NewMockChatSessionServiceInterface(mockCtrl) + + type args struct { + chatSessionID uuid.UUID + } + + tests := []struct { + name string + args args + mockServiceResponseData *domain.ChatSession + mockServiceResponseError error + expected string + expectedStatusCode int + }{ + { + name: "valid", + args: args{ + chatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + }, + mockServiceResponseData: &domain.ChatSession{ + ID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + Title: "title", + }, + mockServiceResponseError: nil, + expected: `{"id":"32345678-0000-0000-0000-000000000000","title":"title","createdAt":"0001-01-01 00:00:00 +0000 UTC","updatedAt":"0001-01-01 00:00:00 +0000 UTC"} +`, + expectedStatusCode: 200, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + mockRequest := httptest.NewRequest( + "GET", + "/chat-sessions/"+tt.args.chatSessionID.String(), + nil, + ) + vars := map[string]string{ + "session_id": tt.args.chatSessionID.String(), + } + mockRequest = mux.SetURLVars(mockRequest, vars) + mockResponseRecorder := httptest.NewRecorder() + + mockService.EXPECT().GetChatSession( + gomock.Any(), + tt.args.chatSessionID, + ).Return(tt.mockServiceResponseData, tt.mockServiceResponseError) + + handler := &GetChatSessionHandler{ + ChatSessionService: mockService, + logger: logger, + } + sut := handler.GetChatSessionController + + sut(mockResponseRecorder, mockRequest) + + mockResponse := mockResponseRecorder.Result() + actual, err := io.ReadAll(mockResponse.Body) + if err != nil { + t.Errorf("error with response reading: %v", err) + return + } + actualStatusCode := mockResponse.StatusCode + + assert.Equal(t, tt.expected, string(actual)) + assert.Equal(t, tt.expectedStatusCode, actualStatusCode) + }) + } +} + +func TestGetChatSessionHandler_GetUserChatSessionsController(t *testing.T) { + logger := logger.NewLogger(context.Background()) + mockCtrl := gomock.NewController(t) + defer mockCtrl.Finish() + + mockService := mock_services.NewMockChatSessionServiceInterface(mockCtrl) + + type args struct { + userID uuid.UUID + } + + tests := []struct { + name string + args args + mockServiceResponseData []*domain.ChatSession + mockServiceResponseError error + expected string + expectedStatusCode int + }{ + { + name: "valid", + args: args{ + userID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + }, + mockServiceResponseData: []*domain.ChatSession{ + &domain.ChatSession{ + ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + Title: "title1", + }, + &domain.ChatSession{ + ID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + Title: "title2", + }, + }, + mockServiceResponseError: nil, + expected: `{"sessions":[{"id":"12345678-0000-0000-0000-000000000000","title":"title1","createdAt":"0001-01-01 00:00:00 +0000 UTC","updatedAt":"0001-01-01 00:00:00 +0000 UTC"},{"id":"32345678-0000-0000-0000-000000000000","title":"title2","createdAt":"0001-01-01 00:00:00 +0000 UTC","updatedAt":"0001-01-01 00:00:00 +0000 UTC"}]} +`, + expectedStatusCode: 200, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + mockRequest := httptest.NewRequest( + "GET", + "/user/"+tt.args.userID.String()+"/chat-sessions", + nil, + ) + vars := map[string]string{ + "user_id": tt.args.userID.String(), + } + mockRequest = mux.SetURLVars(mockRequest, vars) + mockResponseRecorder := httptest.NewRecorder() + + mockService.EXPECT().GetUserChatSessions( + gomock.Any(), + tt.args.userID, + ).Return(tt.mockServiceResponseData, tt.mockServiceResponseError) + + handler := &GetChatSessionHandler{ + ChatSessionService: mockService, + logger: logger, + } + sut := handler.GetUserChatSessionsController + + sut(mockResponseRecorder, mockRequest) + + mockResponse := mockResponseRecorder.Result() + actual, err := io.ReadAll(mockResponse.Body) + if err != nil { + t.Errorf("error with response reading: %v", err) + return + } + actualStatusCode := mockResponse.StatusCode + + assert.Equal(t, tt.expected, string(actual)) + assert.Equal(t, tt.expectedStatusCode, actualStatusCode) + }) + } +} diff --git a/internal/repositories/chatSessionDao.go b/internal/repositories/chatSessionDao.go index 8ab3531..8dd6d40 100644 --- a/internal/repositories/chatSessionDao.go +++ b/internal/repositories/chatSessionDao.go @@ -11,13 +11,13 @@ type ChatSession struct { Title string `gorm:"type:text"` CreatedAt time.Time `gorm:"autoCreateTime"` UpdatedAt time.Time `gorm:"autoUpdateTime"` - Messages []Message `gorm:"foreignKey:SessionID;constraint:OnDelete:CASCADE"` + Messages []Message } type Message struct { - ID uuid.UUID `gorm:"type:uuid;primaryKey;default:uuid_generate_v4()"` - SessionID uuid.UUID `gorm:"type:uuid;not null;index"` - Sender string `gorm:"type:text;not null"` - Content string `gorm:"type:text;not null"` - Timestamp time.Time `gorm:"not null"` + ID uuid.UUID `gorm:"type:uuid;primaryKey;default:uuid_generate_v4()"` + ChatSessionID uuid.UUID `gorm:"type:uuid;not null;index"` + Sender string `gorm:"type:text;not null"` + Content string `gorm:"type:text;not null"` + Timestamp time.Time `gorm:"not null"` } diff --git a/internal/repositories/chatSessionRepositoryGet.go b/internal/repositories/chatSessionRepositoryGet.go new file mode 100644 index 0000000..3a2dc14 --- /dev/null +++ b/internal/repositories/chatSessionRepositoryGet.go @@ -0,0 +1,107 @@ +package repositories + +import ( + "context" + "errors" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "gorm.io/gorm" +) + +func (repo *ChatSessionRepository) GetChatSession( + ctx context.Context, + uuid uuid.UUID, +) (*domain.ChatSession, error) { + var err error + var modelChatSession *ChatSession + + err = repo.db.WithContext(ctx). + Preload("Messages"). + Model(ChatSession{}). + Where("id = ?", uuid). + Take(&modelChatSession).Error + + if err == gorm.ErrRecordNotFound { + return &domain.ChatSession{}, apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("chatSessionID " + uuid.String() + " not found"), + } + } + + if err != nil { + return &domain.ChatSession{}, err + } + + messages := make([]*domain.Message, len(modelChatSession.Messages)) + for i, msg := range modelChatSession.Messages { + messages[i] = &domain.Message{ + ID: msg.ID, + ChatSessionID: msg.ChatSessionID, + Sender: msg.Sender, + Content: msg.Content, + Timestamp: msg.Timestamp, + } + } + + return &domain.ChatSession{ + ID: modelChatSession.ID, + Title: modelChatSession.Title, + UserID: modelChatSession.UserID, + CreatedAt: modelChatSession.CreatedAt, + UpdatedAt: modelChatSession.UpdatedAt, + Messages: messages, + }, err +} + +func (repo *ChatSessionRepository) GetUserChatSessions( + ctx context.Context, + uuid uuid.UUID, +) ([]*domain.ChatSession, error) { + var err error + var modelChatSessions []*ChatSession + + err = repo.db.WithContext(ctx). + Preload("Messages"). + Model(ChatSession{}). + Where("user_id = ?", uuid). + Find(&modelChatSessions).Error + + if err == gorm.ErrRecordNotFound { + return []*domain.ChatSession{}, apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("user uuid " + uuid.String() + " not found"), + } + } + + if err != nil { + return []*domain.ChatSession{}, err + } + + chatSessions := make([]*domain.ChatSession, 0, len(modelChatSessions)) + for _, modelChatSession := range modelChatSessions { + + messages := make([]*domain.Message, len(modelChatSession.Messages)) + for i, msg := range modelChatSession.Messages { + messages[i] = &domain.Message{ + ID: msg.ID, + ChatSessionID: msg.ChatSessionID, + Sender: msg.Sender, + Content: msg.Content, + Timestamp: msg.Timestamp, + } + } + + chatSessions = append( + chatSessions, + &domain.ChatSession{ + ID: modelChatSession.ID, + Title: modelChatSession.Title, + UserID: modelChatSession.UserID, + CreatedAt: modelChatSession.CreatedAt, + UpdatedAt: modelChatSession.UpdatedAt, + Messages: messages, + }, + ) + } + + return chatSessions, nil +} diff --git a/internal/repositories/chatSessionRepositoryGet_test.go b/internal/repositories/chatSessionRepositoryGet_test.go new file mode 100644 index 0000000..d85bf65 --- /dev/null +++ b/internal/repositories/chatSessionRepositoryGet_test.go @@ -0,0 +1,378 @@ +package repositories + +import ( + "context" + "errors" + "github.com/DATA-DOG/go-sqlmock" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/stretchr/testify/assert" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "regexp" + "testing" + "time" +) + +func TestChatRepository_GetChatSession(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + uuid uuid.UUID + } + tests := []struct { + name string + args args + mockSqlChatQueryExpected string + mockSqlMessagesQueryExpected string + mockChatReturned *ChatSession + mockMessagesReturned []*Message + expected *domain.ChatSession + }{ + { + name: "valid", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + mockSqlChatQueryExpected: `SELECT * FROM "chat_sessions" WHERE id = $1 LIMIT $2`, + mockSqlMessagesQueryExpected: `SELECT * FROM "messages" WHERE "messages"."chat_session_id" = $1`, + mockChatReturned: &ChatSession{ + ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + Title: "mockTitle", + CreatedAt: time.Time{}, + UpdatedAt: time.Time{}, + }, + mockMessagesReturned: []*Message{ + { + ID: uuid.UUID{0x02, 0x34, 0x56, 0x68}, + ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + Sender: "SYSTEM", + Content: "MAY THE FORCE BE WITH YOU", + Timestamp: time.Time{}, + }, + { + ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, + ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + Sender: "USER", + Content: "2MAY THE FORCE BE WITH YOU2", + Timestamp: time.Time{}, + }, + }, + expected: &domain.ChatSession{ + ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + Title: "mockTitle", + CreatedAt: time.Time{}, + UpdatedAt: time.Time{}, + Messages: []*domain.Message{ + { + ID: uuid.UUID{0x02, 0x34, 0x56, 0x68}, + ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + Sender: "SYSTEM", + Content: "MAY THE FORCE BE WITH YOU", + Timestamp: time.Time{}, + }, + { + ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, + ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + Sender: "USER", + Content: "2MAY THE FORCE BE WITH YOU2", + Timestamp: time.Time{}, + }, + }, + }, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &ChatSessionRepository{ + db: gormDb, + } + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlChatQueryExpected)). + WithArgs(tt.args.uuid, 1). + WillReturnRows( + sqlmock.NewRows( + []string{"id", "title", "user_id", "created_at", "updated_at"}, + ).AddRow( + tt.mockChatReturned.ID, tt.mockChatReturned.Title, tt.mockChatReturned.UserID, tt.mockChatReturned.CreatedAt, tt.mockChatReturned.UpdatedAt, + ), + ) + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlMessagesQueryExpected)). + WithArgs(tt.args.uuid). + WillReturnRows( + sqlmock.NewRows( + []string{"id", "chat_session_id", "sender", "timestamp", "content"}, + ).AddRow( + tt.mockMessagesReturned[0].ID, tt.mockMessagesReturned[0].ChatSessionID, + tt.mockMessagesReturned[0].Sender, tt.mockMessagesReturned[0].Timestamp, tt.mockMessagesReturned[0].Content, + ).AddRow( + tt.mockMessagesReturned[1].ID, tt.mockMessagesReturned[1].ChatSessionID, + tt.mockMessagesReturned[1].Sender, tt.mockMessagesReturned[1].Timestamp, tt.mockMessagesReturned[1].Content, + ), + ) + + actual, err := repo.GetChatSession(context.Background(), tt.args.uuid) + if err != nil { + t.Errorf("GetChatSession() error = %v", err) + return + } + + assert.Equal(t, tt.expected, actual) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} + +func TestChatRepository_GetChatSessionWithError(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + uuid uuid.UUID + } + tests := []struct { + name string + args args + mockSqlChatQueryExpected string + mockSqlErrorReturned error + expectedError error + }{ + { + name: "random error", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + mockSqlChatQueryExpected: `SELECT * FROM "chat_sessions" WHERE id = $1 LIMIT $2`, + mockSqlErrorReturned: errors.New("random error"), + expectedError: errors.New("random error"), + }, + { + name: "session not found", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + mockSqlChatQueryExpected: `SELECT * FROM "chat_sessions" WHERE id = $1 LIMIT $2`, + mockSqlErrorReturned: gorm.ErrRecordNotFound, + expectedError: apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("chatSessionID 12345678-0000-0000-0000-000000000000 not found"), + }, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &ChatSessionRepository{ + db: gormDb, + } + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlChatQueryExpected)). + WithArgs(tt.args.uuid, 1). + WillReturnError(tt.mockSqlErrorReturned) + + _, actual := repo.GetChatSession(context.Background(), tt.args.uuid) + + assert.Equal(t, actual, tt.expectedError) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} + +func TestChatRepository_GetUserChatSessions(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + uuid uuid.UUID + } + tests := []struct { + name string + args args + mockSqlChatQueryExpected string + mockSqlMessagesQueryExpected string + mockChatsReturned []*ChatSession + mockMessagesReturned []*Message + expected []*domain.ChatSession + }{ + { + name: "valid", + args: args{ + uuid: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + }, + mockSqlChatQueryExpected: `SELECT * FROM "chat_sessions" WHERE user_id = $1`, + mockSqlMessagesQueryExpected: `SELECT * FROM "messages" WHERE "messages"."chat_session_id" IN ($1,$2)`, + mockChatsReturned: []*ChatSession{ + &ChatSession{ + ID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + Title: "mockTitle", + CreatedAt: time.Time{}, + UpdatedAt: time.Time{}, + }, + &ChatSession{ + ID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + Title: "mockTitle", + CreatedAt: time.Time{}, + UpdatedAt: time.Time{}, + }, + }, + mockMessagesReturned: []*Message{ + + { + ID: uuid.UUID{0x02, 0x34, 0x56, 0x68}, + ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + Sender: "SYSTEM", + Content: "MAY THE FORCE BE WITH YOU", + Timestamp: time.Time{}, + }, + { + ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, + ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + Sender: "USER", + Content: "2MAY THE FORCE BE WITH YOU2", + Timestamp: time.Time{}, + }, + + { + ID: uuid.UUID{0x02, 0x34, 0x56, 0x68}, + ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + Sender: "SYSTEM", + Content: "3MAY THE FORCE BE WITH YOU3", + Timestamp: time.Time{}, + }, + { + ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, + ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + Sender: "USER", + Content: "4MAY THE FORCE BE WITH YOU4", + Timestamp: time.Time{}, + }, + }, + expected: []*domain.ChatSession{ + &domain.ChatSession{ + ID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + Title: "mockTitle", + CreatedAt: time.Time{}, + UpdatedAt: time.Time{}, + Messages: []*domain.Message{ + { + ID: uuid.UUID{0x02, 0x34, 0x56, 0x68}, + ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + Sender: "SYSTEM", + Content: "MAY THE FORCE BE WITH YOU", + Timestamp: time.Time{}, + }, + { + + ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, + ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + Sender: "USER", + Content: "2MAY THE FORCE BE WITH YOU2", + Timestamp: time.Time{}, + }, + }, + }, + &domain.ChatSession{ + ID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + UserID: uuid.UUID{0x22, 0x34, 0x56, 0x88}, + Title: "mockTitle", + CreatedAt: time.Time{}, + UpdatedAt: time.Time{}, + Messages: []*domain.Message{ + { + ID: uuid.UUID{0x02, 0x34, 0x56, 0x68}, + ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + Sender: "SYSTEM", + Content: "3MAY THE FORCE BE WITH YOU3", + Timestamp: time.Time{}, + }, + { + ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, + ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + Sender: "USER", + Content: "4MAY THE FORCE BE WITH YOU4", + Timestamp: time.Time{}, + }, + }, + }, + }, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &ChatSessionRepository{ + db: gormDb, + } + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlChatQueryExpected)). + WithArgs(tt.args.uuid). + WillReturnRows( + sqlmock.NewRows( + []string{"id", "title", "user_id", "created_at", "updated_at"}, + ).AddRow( + tt.mockChatsReturned[0].ID, tt.mockChatsReturned[0].Title, tt.mockChatsReturned[0].UserID, tt.mockChatsReturned[0].CreatedAt, tt.mockChatsReturned[0].UpdatedAt, + ).AddRow( + tt.mockChatsReturned[1].ID, tt.mockChatsReturned[1].Title, tt.mockChatsReturned[1].UserID, tt.mockChatsReturned[1].CreatedAt, tt.mockChatsReturned[1].UpdatedAt, + ), + ) + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlMessagesQueryExpected)). + WithArgs(tt.mockChatsReturned[0].ID, tt.mockChatsReturned[1].ID). + WillReturnRows( + sqlmock.NewRows( + []string{"id", "chat_session_id", "sender", "timestamp", "content"}, + ).AddRow( + tt.mockMessagesReturned[0].ID, tt.mockMessagesReturned[0].ChatSessionID, + tt.mockMessagesReturned[0].Sender, tt.mockMessagesReturned[0].Timestamp, tt.mockMessagesReturned[0].Content, + ).AddRow( + tt.mockMessagesReturned[1].ID, tt.mockMessagesReturned[1].ChatSessionID, + tt.mockMessagesReturned[1].Sender, tt.mockMessagesReturned[1].Timestamp, tt.mockMessagesReturned[1].Content, + ).AddRow( + tt.mockMessagesReturned[2].ID, tt.mockMessagesReturned[2].ChatSessionID, + tt.mockMessagesReturned[2].Sender, tt.mockMessagesReturned[2].Timestamp, tt.mockMessagesReturned[2].Content, + ).AddRow( + tt.mockMessagesReturned[3].ID, tt.mockMessagesReturned[3].ChatSessionID, + tt.mockMessagesReturned[3].Sender, tt.mockMessagesReturned[3].Timestamp, tt.mockMessagesReturned[3].Content, + ), + ) + + actual, err := repo.GetUserChatSessions(context.Background(), tt.args.uuid) + if err != nil { + t.Errorf("GetChatSession() error = %v", err) + return + } + + assert.Equal(t, tt.expected, actual) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} diff --git a/mocks/mock_internal/core/services/chatSessionService.go b/mocks/mock_internal/core/services/chatSessionService.go index 3a55e04..3788ecd 100644 --- a/mocks/mock_internal/core/services/chatSessionService.go +++ b/mocks/mock_internal/core/services/chatSessionService.go @@ -85,6 +85,21 @@ func (mr *MockChatSessionServiceInterfaceMockRecorder) GetChatSession(arg0, arg1 return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetChatSession", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).GetChatSession), arg0, arg1) } +// GetUserChatSessions mocks base method. +func (m *MockChatSessionServiceInterface) GetUserChatSessions(arg0 context.Context, arg1 uuid.UUID) ([]*domain.ChatSession, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "GetUserChatSessions", arg0, arg1) + ret0, _ := ret[0].([]*domain.ChatSession) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// GetUserChatSessions indicates an expected call of GetUserChatSessions. +func (mr *MockChatSessionServiceInterfaceMockRecorder) GetUserChatSessions(arg0, arg1 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetUserChatSessions", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).GetUserChatSessions), arg0, arg1) +} + // UpdateChatSessionTitle mocks base method. func (m *MockChatSessionServiceInterface) UpdateChatSessionTitle(arg0 context.Context, arg1 uuid.UUID, arg2 string) error { m.ctrl.T.Helper() diff --git a/pkg/server/routes.go b/pkg/server/routes.go index 641303a..7ccb740 100644 --- a/pkg/server/routes.go +++ b/pkg/server/routes.go @@ -51,7 +51,11 @@ func (s *Server) initializeRoutes() { chatSessionService := services.NewChatSessionService(s.logger, chatSessionRepository) createChatSessionHandler := chatSessions.NewCreateUserChatSessionHandler(chatSessionService, s.logger) + getChatSessionHandler := chatSessions.NewGetChatSessionHandler(chatSessionService, s.logger) protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionAssetController).Methods("POST") + protected.HandleFunc("/users/{user_id}/chat-sessions", getChatSessionHandler.GetUserChatSessionsController).Methods("GET") + + protected.HandleFunc("/chat-sessions/{session_id}", getChatSessionHandler.GetChatSessionController).Methods("GET") } From dadda6fc2bd4fdff66c5fa77510b6ef78a07fc36 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 14:00:18 +0300 Subject: [PATCH 08/52] Small fixes --- internal/core/domain/chatSession.go | 2 +- ...Handler.go => createChatSessionHandler.go} | 2 +- ...st.go => createChatSessionHandler_test.go} | 6 +-- internal/handlers/chatSessions/dto.go | 51 +++++++++++-------- ...getHandler.go => getChatSessionHandler.go} | 0 ..._test.go => getChatSessionHandler_test.go} | 0 internal/repositories/chatSessionDao.go | 2 +- .../chatSessionRepositoryCreate.go | 2 +- .../repositories/chatSessionRepositoryGet.go | 4 +- .../chatSessionRepositoryGet_test.go | 40 +++++++-------- 10 files changed, 59 insertions(+), 50 deletions(-) rename internal/handlers/chatSessions/{createHandler.go => createChatSessionHandler.go} (98%) rename internal/handlers/chatSessions/{createHandler_test.go => createChatSessionHandler_test.go} (97%) rename internal/handlers/chatSessions/{getHandler.go => getChatSessionHandler.go} (100%) rename internal/handlers/chatSessions/{getHandler_test.go => getChatSessionHandler_test.go} (100%) diff --git a/internal/core/domain/chatSession.go b/internal/core/domain/chatSession.go index dcb3b88..4ddc4a0 100644 --- a/internal/core/domain/chatSession.go +++ b/internal/core/domain/chatSession.go @@ -19,5 +19,5 @@ type Message struct { ChatSessionID uuid.UUID Sender string Content string - Timestamp time.Time + CreatedAt time.Time } diff --git a/internal/handlers/chatSessions/createHandler.go b/internal/handlers/chatSessions/createChatSessionHandler.go similarity index 98% rename from internal/handlers/chatSessions/createHandler.go rename to internal/handlers/chatSessions/createChatSessionHandler.go index 4f8bb89..4292fe0 100644 --- a/internal/handlers/chatSessions/createHandler.go +++ b/internal/handlers/chatSessions/createChatSessionHandler.go @@ -35,7 +35,7 @@ func NewCreateUserChatSessionHandler( // @Failure 401 {object} ChatSessionResponse "Authentication error" // @Failure 500 {object} ChatSessionResponse "Internal Server Error" // @Router /users/user_id/chat-sessions [post] -func (handler *CreateUserChatSessionHandler) CreateUserChatSessionAssetController(w http.ResponseWriter, r *http.Request) { +func (handler *CreateUserChatSessionHandler) CreateUserChatSessionController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") ctx := r.Context() diff --git a/internal/handlers/chatSessions/createHandler_test.go b/internal/handlers/chatSessions/createChatSessionHandler_test.go similarity index 97% rename from internal/handlers/chatSessions/createHandler_test.go rename to internal/handlers/chatSessions/createChatSessionHandler_test.go index 2e2c738..90099d7 100644 --- a/internal/handlers/chatSessions/createHandler_test.go +++ b/internal/handlers/chatSessions/createChatSessionHandler_test.go @@ -72,7 +72,7 @@ func TestCreateUserChatSessionHandler_CreateUserChatSessionController(t *testing ChatSessionService: mockService, logger: logger, } - sut := handler.CreateUserChatSessionAssetController + sut := handler.CreateUserChatSessionController sut(mockResponseRecorder, mockRequest) @@ -151,7 +151,7 @@ func TestCreateUserChatSessionHandler_CreateUserChatSessionAssetControllerHasBad ChatSessionService: mockService, logger: logger, } - sut := handler.CreateUserChatSessionAssetController + sut := handler.CreateUserChatSessionController sut(mockResponseRecorder, mockRequest) @@ -239,7 +239,7 @@ func TestCreateUserChatSessionHandler_CreateUserChatSessionAssetControllerHasSer ChatSessionService: mockService, logger: logger, } - sut := handler.CreateUserChatSessionAssetController + sut := handler.CreateUserChatSessionController sut(mockResponseRecorder, mockRequest) diff --git a/internal/handlers/chatSessions/dto.go b/internal/handlers/chatSessions/dto.go index 8fd2759..ac3cfae 100644 --- a/internal/handlers/chatSessions/dto.go +++ b/internal/handlers/chatSessions/dto.go @@ -10,22 +10,22 @@ type UserChatSessionsResponse struct { } type ChatSessionResponse struct { - ID string `json:"id,omitempty"` - Title string `json:"title,omitempty"` - CreatedAt string `json:"createdAt,omitempty"` - UpdatedAt string `json:"updatedAt,omitempty"` - Messages []Message `json:"messages,omitempty"` - ErrorMessage string `json:"errorMessage,omitempty"` + ID string `json:"id,omitempty"` + Title string `json:"title,omitempty"` + CreatedAt string `json:"createdAt,omitempty"` + UpdatedAt string `json:"updatedAt,omitempty"` + Messages []MessageResponse `json:"messages,omitempty"` + ErrorMessage string `json:"errorMessage,omitempty"` } func ChatSessionResponseFromModel(domainChatSession *domain.ChatSession) *ChatSessionResponse { - messages := make([]Message, len(domainChatSession.Messages)) + messages := make([]MessageResponse, len(domainChatSession.Messages)) for i, msg := range domainChatSession.Messages { - messages[i] = Message{ + messages[i] = MessageResponse{ ID: msg.ID.String(), Sender: msg.Sender, Content: msg.Content, - Timestamp: msg.Timestamp.String(), + CreatedAt: msg.CreatedAt.String(), } } @@ -49,11 +49,21 @@ func UserChatSessionsResponseFromModel(domainChatSession []*domain.ChatSession) } } -type Message struct { - ID string `json:"id"` - Sender string `json:"sender" enum:"USER,SYSTEM" example:"USER"` - Content string `json:"content"` - Timestamp string `json:"timestamp"` +type MessageResponse struct { + ID string `json:"id,omitempty"` + Sender string `json:"sender,omitempty" enum:"USER,SYSTEM"` + Content string `json:"content,omitempty"` + CreatedAt string `json:"created_at,omitempty"` + ErrorMessage string `json:"errorMessage,omitempty"` +} + +func MessageResponseFromModel(msg *domain.Message) *MessageResponse { + return &MessageResponse{ + ID: msg.ID.String(), + Sender: msg.Sender, + Content: msg.Content, + CreatedAt: msg.CreatedAt.String(), + } } //type CreateChatSessionRequest struct { @@ -61,14 +71,13 @@ type Message struct { //} type SendMessageRequest struct { - UserID string `json:"chatSessionID"` - SessionID string `json:"sessionId"` - Content string `json:"content"` + //UserID string `json:"chatSessionID"` + //SessionID string `json:"sessionId"` + Content string `json:"content"` } type SendMessageResponse struct { - Body struct { - UserMessage Message `json:"userMessage"` - SystemMessage Message `json:"systemMessage"` - } + UserMessage *MessageResponse `json:"userMessage,omitempty"` + SystemMessage *MessageResponse `json:"systemMessage,omitempty"` + ErrorMessage string `json:"errorMessage,omitempty"` } diff --git a/internal/handlers/chatSessions/getHandler.go b/internal/handlers/chatSessions/getChatSessionHandler.go similarity index 100% rename from internal/handlers/chatSessions/getHandler.go rename to internal/handlers/chatSessions/getChatSessionHandler.go diff --git a/internal/handlers/chatSessions/getHandler_test.go b/internal/handlers/chatSessions/getChatSessionHandler_test.go similarity index 100% rename from internal/handlers/chatSessions/getHandler_test.go rename to internal/handlers/chatSessions/getChatSessionHandler_test.go diff --git a/internal/repositories/chatSessionDao.go b/internal/repositories/chatSessionDao.go index 8dd6d40..cd6da33 100644 --- a/internal/repositories/chatSessionDao.go +++ b/internal/repositories/chatSessionDao.go @@ -19,5 +19,5 @@ type Message struct { ChatSessionID uuid.UUID `gorm:"type:uuid;not null;index"` Sender string `gorm:"type:text;not null"` Content string `gorm:"type:text;not null"` - Timestamp time.Time `gorm:"not null"` + CreatedAt time.Time `gorm:"not null"` } diff --git a/internal/repositories/chatSessionRepositoryCreate.go b/internal/repositories/chatSessionRepositoryCreate.go index 40c450d..cdb7ab5 100644 --- a/internal/repositories/chatSessionRepositoryCreate.go +++ b/internal/repositories/chatSessionRepositoryCreate.go @@ -29,7 +29,7 @@ func (repo *ChatSessionRepository) CreateChatSession( err = repo.db.WithContext(ctx).Create(&modelChat).Error if err != nil { - return uuid.UUID{}, err + return uuid.Nil, err } return modelChat.ID, nil diff --git a/internal/repositories/chatSessionRepositoryGet.go b/internal/repositories/chatSessionRepositoryGet.go index 3a2dc14..b3d5447 100644 --- a/internal/repositories/chatSessionRepositoryGet.go +++ b/internal/repositories/chatSessionRepositoryGet.go @@ -39,7 +39,7 @@ func (repo *ChatSessionRepository) GetChatSession( ChatSessionID: msg.ChatSessionID, Sender: msg.Sender, Content: msg.Content, - Timestamp: msg.Timestamp, + CreatedAt: msg.CreatedAt, } } @@ -86,7 +86,7 @@ func (repo *ChatSessionRepository) GetUserChatSessions( ChatSessionID: msg.ChatSessionID, Sender: msg.Sender, Content: msg.Content, - Timestamp: msg.Timestamp, + CreatedAt: msg.CreatedAt, } } diff --git a/internal/repositories/chatSessionRepositoryGet_test.go b/internal/repositories/chatSessionRepositoryGet_test.go index d85bf65..a7a9664 100644 --- a/internal/repositories/chatSessionRepositoryGet_test.go +++ b/internal/repositories/chatSessionRepositoryGet_test.go @@ -56,14 +56,14 @@ func TestChatRepository_GetChatSession(t *testing.T) { ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, Sender: "SYSTEM", Content: "MAY THE FORCE BE WITH YOU", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, { ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, Sender: "USER", Content: "2MAY THE FORCE BE WITH YOU2", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, }, expected: &domain.ChatSession{ @@ -78,14 +78,14 @@ func TestChatRepository_GetChatSession(t *testing.T) { ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, Sender: "SYSTEM", Content: "MAY THE FORCE BE WITH YOU", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, { ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, Sender: "USER", Content: "2MAY THE FORCE BE WITH YOU2", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, }, }, @@ -111,13 +111,13 @@ func TestChatRepository_GetChatSession(t *testing.T) { WithArgs(tt.args.uuid). WillReturnRows( sqlmock.NewRows( - []string{"id", "chat_session_id", "sender", "timestamp", "content"}, + []string{"id", "chat_session_id", "sender", "created_at", "content"}, ).AddRow( tt.mockMessagesReturned[0].ID, tt.mockMessagesReturned[0].ChatSessionID, - tt.mockMessagesReturned[0].Sender, tt.mockMessagesReturned[0].Timestamp, tt.mockMessagesReturned[0].Content, + tt.mockMessagesReturned[0].Sender, tt.mockMessagesReturned[0].CreatedAt, tt.mockMessagesReturned[0].Content, ).AddRow( tt.mockMessagesReturned[1].ID, tt.mockMessagesReturned[1].ChatSessionID, - tt.mockMessagesReturned[1].Sender, tt.mockMessagesReturned[1].Timestamp, tt.mockMessagesReturned[1].Content, + tt.mockMessagesReturned[1].Sender, tt.mockMessagesReturned[1].CreatedAt, tt.mockMessagesReturned[1].Content, ), ) @@ -248,14 +248,14 @@ func TestChatRepository_GetUserChatSessions(t *testing.T) { ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, Sender: "SYSTEM", Content: "MAY THE FORCE BE WITH YOU", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, { ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, Sender: "USER", Content: "2MAY THE FORCE BE WITH YOU2", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, { @@ -263,14 +263,14 @@ func TestChatRepository_GetUserChatSessions(t *testing.T) { ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, Sender: "SYSTEM", Content: "3MAY THE FORCE BE WITH YOU3", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, { ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, Sender: "USER", Content: "4MAY THE FORCE BE WITH YOU4", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, }, expected: []*domain.ChatSession{ @@ -286,7 +286,7 @@ func TestChatRepository_GetUserChatSessions(t *testing.T) { ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, Sender: "SYSTEM", Content: "MAY THE FORCE BE WITH YOU", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, { @@ -294,7 +294,7 @@ func TestChatRepository_GetUserChatSessions(t *testing.T) { ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, Sender: "USER", Content: "2MAY THE FORCE BE WITH YOU2", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, }, }, @@ -310,14 +310,14 @@ func TestChatRepository_GetUserChatSessions(t *testing.T) { ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, Sender: "SYSTEM", Content: "3MAY THE FORCE BE WITH YOU3", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, { ID: uuid.UUID{0x052, 0x34, 0x56, 0x58}, ChatSessionID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, Sender: "USER", Content: "4MAY THE FORCE BE WITH YOU4", - Timestamp: time.Time{}, + CreatedAt: time.Time{}, }, }, }, @@ -346,19 +346,19 @@ func TestChatRepository_GetUserChatSessions(t *testing.T) { WithArgs(tt.mockChatsReturned[0].ID, tt.mockChatsReturned[1].ID). WillReturnRows( sqlmock.NewRows( - []string{"id", "chat_session_id", "sender", "timestamp", "content"}, + []string{"id", "chat_session_id", "sender", "created_at", "content"}, ).AddRow( tt.mockMessagesReturned[0].ID, tt.mockMessagesReturned[0].ChatSessionID, - tt.mockMessagesReturned[0].Sender, tt.mockMessagesReturned[0].Timestamp, tt.mockMessagesReturned[0].Content, + tt.mockMessagesReturned[0].Sender, tt.mockMessagesReturned[0].CreatedAt, tt.mockMessagesReturned[0].Content, ).AddRow( tt.mockMessagesReturned[1].ID, tt.mockMessagesReturned[1].ChatSessionID, - tt.mockMessagesReturned[1].Sender, tt.mockMessagesReturned[1].Timestamp, tt.mockMessagesReturned[1].Content, + tt.mockMessagesReturned[1].Sender, tt.mockMessagesReturned[1].CreatedAt, tt.mockMessagesReturned[1].Content, ).AddRow( tt.mockMessagesReturned[2].ID, tt.mockMessagesReturned[2].ChatSessionID, - tt.mockMessagesReturned[2].Sender, tt.mockMessagesReturned[2].Timestamp, tt.mockMessagesReturned[2].Content, + tt.mockMessagesReturned[2].Sender, tt.mockMessagesReturned[2].CreatedAt, tt.mockMessagesReturned[2].Content, ).AddRow( tt.mockMessagesReturned[3].ID, tt.mockMessagesReturned[3].ChatSessionID, - tt.mockMessagesReturned[3].Sender, tt.mockMessagesReturned[3].Timestamp, tt.mockMessagesReturned[3].Content, + tt.mockMessagesReturned[3].Sender, tt.mockMessagesReturned[3].CreatedAt, tt.mockMessagesReturned[3].Content, ), ) From a6461f9b68115d6e82fc8090b82780213eb341ff Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 14:00:49 +0300 Subject: [PATCH 09/52] Adds send message functionality --- .../core/ports/messageRepositoryInterface.go | 12 + internal/core/services/messageService.go | 70 ++++++ .../chatSessions/sendMessageHandler.go | 206 ++++++++++++++++++ .../chatSessions/sendMessageHandler_test.go | 114 ++++++++++ .../repositories/messageRepositoryCreate.go | 37 ++++ .../messageRepositoryCreate_test.go | 78 +++++++ internal/repositories/messageRepositoryGet.go | 41 ++++ .../repositories/messageRepositoryGet_test.go | 83 +++++++ .../core/services/messageService.go | 72 ++++++ pkg/errors/errors.go | 16 ++ pkg/server/routes.go | 6 +- 11 files changed, 734 insertions(+), 1 deletion(-) create mode 100644 internal/core/ports/messageRepositoryInterface.go create mode 100644 internal/core/services/messageService.go create mode 100644 internal/handlers/chatSessions/sendMessageHandler.go create mode 100644 internal/handlers/chatSessions/sendMessageHandler_test.go create mode 100644 internal/repositories/messageRepositoryCreate.go create mode 100644 internal/repositories/messageRepositoryCreate_test.go create mode 100644 internal/repositories/messageRepositoryGet.go create mode 100644 internal/repositories/messageRepositoryGet_test.go create mode 100644 mocks/mock_internal/core/services/messageService.go diff --git a/internal/core/ports/messageRepositoryInterface.go b/internal/core/ports/messageRepositoryInterface.go new file mode 100644 index 0000000..10905e3 --- /dev/null +++ b/internal/core/ports/messageRepositoryInterface.go @@ -0,0 +1,12 @@ +package ports + +import ( + "context" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" +) + +type MessageRepositoryInterface interface { + CreateMessage(context.Context, *domain.Message) (uuid.UUID, error) + GetMessage(context.Context, uuid.UUID) (*domain.Message, error) +} diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go new file mode 100644 index 0000000..01e7151 --- /dev/null +++ b/internal/core/services/messageService.go @@ -0,0 +1,70 @@ +package services + +import ( + "context" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/ports" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" +) + +type MessageServiceInterface interface { + CreateMessage(context.Context, uuid.UUID, *domain.Message) (uuid.UUID, error) + GetAnswerForMessage(context.Context, uuid.UUID) (*domain.Message, error) +} + +type MessageService struct { + logger logger.LoggerInterface + messageRepository ports.MessageRepositoryInterface + chatSessionRepository ports.ChatSessionRepositoryInterface +} + +func NewMessageService( + logger logger.LoggerInterface, + messageRepositoryInterface ports.MessageRepositoryInterface, + chatSessionRepository ports.ChatSessionRepositoryInterface, +) *MessageService { + return &MessageService{ + logger: logger, + messageRepository: messageRepositoryInterface, + chatSessionRepository: chatSessionRepository, + } +} + +func (s MessageService) CreateMessage(ctx context.Context, userID uuid.UUID, message *domain.Message) (uuid.UUID, error) { + chatSession, err := s.chatSessionRepository.GetChatSession(ctx, message.ChatSessionID) + if err != nil { + return uuid.Nil, err + } + + if chatSession.UserID != userID { + return uuid.Nil, apierrors.NewUserMismatchError(message.ChatSessionID.String(), userID.String()) + } + + return s.messageRepository.CreateMessage(ctx, message) +} + +func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageID uuid.UUID) (*domain.Message, error) { + initialMessage, err := s.messageRepository.GetMessage(ctx, initialMessageID) + if err != nil { + return nil, err + } + + replyMessage := &domain.Message{ + ID: initialMessageID, + ChatSessionID: initialMessage.ChatSessionID, + Content: "This is a mock response from the LLM", + Sender: "SYSTEM", //TODO: constant + } + + //TODO: get from llm + insertedMessageID, err := s.messageRepository.CreateMessage(ctx, replyMessage) + if err != nil { + return nil, err + } + + replyMessage.ID = insertedMessageID + + return replyMessage, nil +} diff --git a/internal/handlers/chatSessions/sendMessageHandler.go b/internal/handlers/chatSessions/sendMessageHandler.go new file mode 100644 index 0000000..e03b1e9 --- /dev/null +++ b/internal/handlers/chatSessions/sendMessageHandler.go @@ -0,0 +1,206 @@ +package chatSessions + +import ( + "encoding/json" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" +) + +type SendMessageHandler struct { + MessageService services.MessageServiceInterface + logger logger.LoggerInterface +} + +func NewSendMessageHandler( + service services.MessageServiceInterface, + logger logger.LoggerInterface, +) *SendMessageHandler { + return &SendMessageHandler{ + MessageService: service, + logger: logger, + } +} + +// @Summary Sends message to a given chat session and gets response +// @Description Sends message to a given chat session and gets response +// @Security BearerAuth +// @Param SendMessageRequest body SendMessageRequest true "request body" +// @Param user_id path int true "user id" +// @Param session_id body int true "session id" +// @Success 201 {object} MessageResponse +// @Failure 400 {object} MessageResponse "Error in message payload" +// @Failure 401 {object} MessageResponse "Authentication error" +// @Failure 500 {object} MessageResponse "Internal Server Error" +// @Router /users/user_id/chat-sessions/session_id/messages [post] +func (handler *SendMessageHandler) SendMessageController(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + + ctx := r.Context() + + var err error + response := &SendMessageResponse{} + request := &SendMessageRequest{} + + userIdAsString := mux.Vars(r)["user_id"] + if userIdAsString == "" { + response.ErrorMessage = "missing user id" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + userId, err := uuid.Parse(userIdAsString) + if err != nil { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed user uuid" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + chatSessionIDAsString := mux.Vars(r)["session_id"] + if chatSessionIDAsString == "" { + response.ErrorMessage = "missing session id" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + chatSessionID, err := uuid.Parse(chatSessionIDAsString) + if err != nil { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed session uuid" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + err = json.NewDecoder(r.Body).Decode(request) + if err != nil { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed sending message request" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + domainMessage := &domain.Message{ + ChatSessionID: chatSessionID, + Content: request.Content, + Sender: "USER", //TODO: hardcode for now, should be set by the user + } + + insertedUUID, err := handler.MessageService.CreateMessage( + ctx, + userId, + domainMessage, + ) + + if resourceNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": resourceNotFound.Unwrap(), + }) + + response.ErrorMessage = err.Error() + handler.JsonResponse(w, http.StatusNotFound, response) + + return + } + + if userMismatchError, ok := err.(apierrors.UserMismatchError); ok { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": userMismatchError.Error(), + }) + + response.ErrorMessage = err.Error() + handler.JsonResponse(w, http.StatusForbidden, response) + + return + } + + if err != nil { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "error in sending message" + handler.JsonResponse(w, http.StatusInternalServerError, response) + + return + } + + domainMessage.ID = insertedUUID + + replyMessage, err := handler.MessageService.GetAnswerForMessage(ctx, insertedUUID) + if resourceNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Error in replying to message", + map[string]interface{}{ + "errorMessage": resourceNotFound.Unwrap(), + }) + + response.ErrorMessage = err.Error() + handler.JsonResponse(w, http.StatusNotFound, response) + + return + } + + if err != nil { + handler.logger.Error("Error in replying to message", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "error in replying to message" + handler.JsonResponse(w, http.StatusInternalServerError, response) + + return + } + + response.UserMessage = MessageResponseFromModel(domainMessage) + response.SystemMessage = MessageResponseFromModel(replyMessage) + + handler.JsonResponse(w, http.StatusOK, response) +} + +func (handler *SendMessageHandler) JsonResponse( + w http.ResponseWriter, + statusCode int, + response *SendMessageResponse, +) { + w.WriteHeader(statusCode) + err := json.NewEncoder(w).Encode(response) + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + response.ErrorMessage = "error in sending message - json response" + + handler.logger.Error("Error in sending message - json response", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + } +} diff --git a/internal/handlers/chatSessions/sendMessageHandler_test.go b/internal/handlers/chatSessions/sendMessageHandler_test.go new file mode 100644 index 0000000..fc40ac7 --- /dev/null +++ b/internal/handlers/chatSessions/sendMessageHandler_test.go @@ -0,0 +1,114 @@ +package chatSessions + +import ( + "bytes" + "context" + "encoding/json" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + mock_services "github.com/loukaspe/jedi-team-challenge/mocks/mock_internal/core/services" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/stretchr/testify/assert" + "go.uber.org/mock/gomock" + "io" + "net/http/httptest" + "testing" + "time" +) + +func TestSendMessageHandler_SendMessageController(t *testing.T) { + logger := logger.NewLogger(context.Background()) + mockCtrl := gomock.NewController(t) + defer mockCtrl.Finish() + + mockMessageService := mock_services.NewMockMessageServiceInterface(mockCtrl) + + type args struct { + userId uuid.UUID + sessionId uuid.UUID + } + + tests := []struct { + name string + args args + mockMessageInsertedID uuid.UUID + mockReplyMessageInserted *domain.Message + expected []byte + expectedStatusCode int + }{ + { + name: "valid", + args: args{ + userId: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + sessionId: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + }, + mockMessageInsertedID: uuid.UUID{0x42, 0x34, 0x56, 0x88}, + mockReplyMessageInserted: &domain.Message{ + ID: uuid.UUID{0x52, 0x34, 0x56, 0x88}, + ChatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + Sender: "SYSTEM", + Content: "Reply", + CreatedAt: time.Time{}, + }, + expected: json.RawMessage(`{"userMessage":{"id":"42345688-0000-0000-0000-000000000000","sender":"USER","content":"Hello, this is a test message","created_at":"0001-01-01 00:00:00 +0000 UTC"},"systemMessage":{"id":"52345688-0000-0000-0000-000000000000","sender":"SYSTEM","content":"Reply","created_at":"0001-01-01 00:00:00 +0000 UTC"}} +`), + expectedStatusCode: 200, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + mockRequest := httptest.NewRequest( + "POST", + "/users/"+tt.args.userId.String()+"/chat-sessions"+"/"+tt.args.sessionId.String()+"/messages", + bytes.NewBuffer( + json.RawMessage(`{"content":"Hello, this is a test message"}`), + ), + ) + + vars := map[string]string{ + "user_id": tt.args.userId.String(), + "session_id": tt.args.sessionId.String(), + } + mockRequest = mux.SetURLVars(mockRequest, vars) + + mockRequest.Header.Set("Content-Type", "application/json") + mockResponseRecorder := httptest.NewRecorder() + + mockMessageService.EXPECT().CreateMessage( + gomock.Any(), + tt.args.userId, + &domain.Message{ + ChatSessionID: tt.args.sessionId, + Sender: "USER", + Content: "Hello, this is a test message", + }, + ).Return(tt.mockMessageInsertedID, nil) + + mockMessageService.EXPECT().GetAnswerForMessage( + gomock.Any(), + tt.mockMessageInsertedID, + ).Return(tt.mockReplyMessageInserted, nil) + + handler := &SendMessageHandler{ + MessageService: mockMessageService, + logger: logger, + } + sut := handler.SendMessageController + + sut(mockResponseRecorder, mockRequest) + + mockResponse := mockResponseRecorder.Result() + actual, err := io.ReadAll(mockResponse.Body) + if err != nil { + t.Errorf("error with response reading: %v", err) + return + } + actualStatusCode := mockResponse.StatusCode + + assert.Equal(t, string(tt.expected), string(actual)) + assert.Equal(t, tt.expectedStatusCode, actualStatusCode) + }) + } +} diff --git a/internal/repositories/messageRepositoryCreate.go b/internal/repositories/messageRepositoryCreate.go new file mode 100644 index 0000000..0894fcb --- /dev/null +++ b/internal/repositories/messageRepositoryCreate.go @@ -0,0 +1,37 @@ +package repositories + +import ( + "context" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "gorm.io/gorm" +) + +type MessageRepository struct { + db *gorm.DB +} + +func NewMessageRepository(db *gorm.DB) *MessageRepository { + return &MessageRepository{db: db} +} + +func (repo *MessageRepository) CreateMessage( + ctx context.Context, + chat *domain.Message, +) (uuid.UUID, error) { + var err error + + modelChat := Message{ + ChatSessionID: chat.ChatSessionID, + Sender: chat.Sender, + Content: chat.Content, + CreatedAt: chat.CreatedAt, + } + + err = repo.db.WithContext(ctx).Create(&modelChat).Error + if err != nil { + return uuid.Nil, err + } + + return modelChat.ID, nil +} diff --git a/internal/repositories/messageRepositoryCreate_test.go b/internal/repositories/messageRepositoryCreate_test.go new file mode 100644 index 0000000..e15fdb3 --- /dev/null +++ b/internal/repositories/messageRepositoryCreate_test.go @@ -0,0 +1,78 @@ +package repositories + +import ( + "context" + "github.com/DATA-DOG/go-sqlmock" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/stretchr/testify/assert" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "regexp" + "testing" + "time" +) + +func TestChatRepository_CreateMessage(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + message *domain.Message + } + tests := []struct { + name string + args args + mockSqlMessageQueryExpected string + mockInsertedMessageIdReturned uuid.UUID + expectedMessageUid uuid.UUID + }{ + { + name: "valid", + args: args{ + message: &domain.Message{ + ChatSessionID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + Sender: "USER", + Content: "ablaabla", + CreatedAt: time.Time{}, + }, + }, + mockSqlMessageQueryExpected: `INSERT INTO "messages" ("chat_session_id","sender","content","created_at") VALUES ($1,$2,$3,$4) RETURNING "id"`, + mockInsertedMessageIdReturned: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + expectedMessageUid: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &MessageRepository{ + db: gormDb, + } + + mockDb.ExpectBegin() + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlMessageQueryExpected)). + WithArgs( + tt.args.message.ChatSessionID, tt.args.message.Sender, tt.args.message.Content, sqlmock.AnyArg(), + ). + WillReturnRows(sqlmock.NewRows([]string{"id"}).AddRow(tt.mockInsertedMessageIdReturned)) + mockDb.ExpectCommit() + + actual, err := repo.CreateMessage(context.Background(), tt.args.message) + if err != nil { + t.Errorf("CreateMessage() error = %v", err) + } + + assert.Equal(t, tt.expectedMessageUid, actual) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} diff --git a/internal/repositories/messageRepositoryGet.go b/internal/repositories/messageRepositoryGet.go new file mode 100644 index 0000000..c2b27d8 --- /dev/null +++ b/internal/repositories/messageRepositoryGet.go @@ -0,0 +1,41 @@ +package repositories + +import ( + "context" + "errors" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "gorm.io/gorm" +) + +func (repo *MessageRepository) GetMessage( + ctx context.Context, + uuid uuid.UUID, +) (*domain.Message, error) { + var err error + var modelMessage *Message + + err = repo.db.WithContext(ctx). + Model(Message{}). + Where("id = ?", uuid). + Take(&modelMessage).Error + + if err == gorm.ErrRecordNotFound { + return &domain.Message{}, apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("messageID " + uuid.String() + " not found"), + } + } + + if err != nil { + return &domain.Message{}, err + } + + return &domain.Message{ + ID: modelMessage.ID, + ChatSessionID: modelMessage.ChatSessionID, + Sender: modelMessage.Sender, + Content: modelMessage.Content, + CreatedAt: modelMessage.CreatedAt, + }, err +} diff --git a/internal/repositories/messageRepositoryGet_test.go b/internal/repositories/messageRepositoryGet_test.go new file mode 100644 index 0000000..d3c6a38 --- /dev/null +++ b/internal/repositories/messageRepositoryGet_test.go @@ -0,0 +1,83 @@ +package repositories + +import ( + "context" + "github.com/DATA-DOG/go-sqlmock" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/stretchr/testify/assert" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "regexp" + "testing" +) + +func TestChatRepository_GetMessage(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + uuid uuid.UUID + } + tests := []struct { + name string + args args + mockSqlMessageQueryExpected string + mockMessageReturned *Message + expected *domain.Message + }{ + { + name: "valid", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + }, + mockSqlMessageQueryExpected: `SELECT * FROM "messages" WHERE id = $1 LIMIT $2`, + mockMessageReturned: &Message{ + ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + ChatSessionID: uuid.UUID{0x22, 0x34, 0x56, 0x78}, + Sender: "user1", + Content: "Hello, world!", + }, + expected: &domain.Message{ + ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + ChatSessionID: uuid.UUID{0x22, 0x34, 0x56, 0x78}, + Sender: "user1", + Content: "Hello, world!", + }, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &MessageRepository{ + db: gormDb, + } + + mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlMessageQueryExpected)). + WithArgs(tt.args.uuid, 1). + WillReturnRows( + sqlmock.NewRows( + []string{"id", "chat_session_id", "sender", "created_at", "content"}, + ).AddRow( + tt.mockMessageReturned.ID, tt.mockMessageReturned.ChatSessionID, + tt.mockMessageReturned.Sender, tt.mockMessageReturned.CreatedAt, tt.mockMessageReturned.Content, + )) + + actual, err := repo.GetMessage(context.Background(), tt.args.uuid) + if err != nil { + t.Errorf("GetMessage() error = %v", err) + return + } + + assert.Equal(t, tt.expected, actual) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} diff --git a/mocks/mock_internal/core/services/messageService.go b/mocks/mock_internal/core/services/messageService.go new file mode 100644 index 0000000..aaa8dff --- /dev/null +++ b/mocks/mock_internal/core/services/messageService.go @@ -0,0 +1,72 @@ +// Code generated by MockGen. DO NOT EDIT. +// Source: ../internal/core/services/messageService.go +// +// Generated by this command: +// +// mockgen -source=../internal/core/services/messageService.go -destination=../mocks/mock_internal/core/services/messageService.go +// + +// Package mock_services is a generated GoMock package. +package mock_services + +import ( + context "context" + reflect "reflect" + + uuid "github.com/google/uuid" + domain "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + gomock "go.uber.org/mock/gomock" +) + +// MockMessageServiceInterface is a mock of MessageServiceInterface interface. +type MockMessageServiceInterface struct { + ctrl *gomock.Controller + recorder *MockMessageServiceInterfaceMockRecorder +} + +// MockMessageServiceInterfaceMockRecorder is the mock recorder for MockMessageServiceInterface. +type MockMessageServiceInterfaceMockRecorder struct { + mock *MockMessageServiceInterface +} + +// NewMockMessageServiceInterface creates a new mock instance. +func NewMockMessageServiceInterface(ctrl *gomock.Controller) *MockMessageServiceInterface { + mock := &MockMessageServiceInterface{ctrl: ctrl} + mock.recorder = &MockMessageServiceInterfaceMockRecorder{mock} + return mock +} + +// EXPECT returns an object that allows the caller to indicate expected use. +func (m *MockMessageServiceInterface) EXPECT() *MockMessageServiceInterfaceMockRecorder { + return m.recorder +} + +// CreateMessage mocks base method. +func (m *MockMessageServiceInterface) CreateMessage(arg0 context.Context, arg1 uuid.UUID, arg2 *domain.Message) (uuid.UUID, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "CreateMessage", arg0, arg1, arg2) + ret0, _ := ret[0].(uuid.UUID) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// CreateMessage indicates an expected call of CreateMessage. +func (mr *MockMessageServiceInterfaceMockRecorder) CreateMessage(arg0, arg1, arg2 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "CreateMessage", reflect.TypeOf((*MockMessageServiceInterface)(nil).CreateMessage), arg0, arg1, arg2) +} + +// GetAnswerForMessage mocks base method. +func (m *MockMessageServiceInterface) GetAnswerForMessage(arg0 context.Context, arg1 uuid.UUID) (*domain.Message, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "GetAnswerForMessage", arg0, arg1) + ret0, _ := ret[0].(*domain.Message) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// GetAnswerForMessage indicates an expected call of GetAnswerForMessage. +func (mr *MockMessageServiceInterfaceMockRecorder) GetAnswerForMessage(arg0, arg1 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetAnswerForMessage", reflect.TypeOf((*MockMessageServiceInterface)(nil).GetAnswerForMessage), arg0, arg1) +} diff --git a/pkg/errors/errors.go b/pkg/errors/errors.go index b90e9a7..69a0943 100644 --- a/pkg/errors/errors.go +++ b/pkg/errors/errors.go @@ -13,3 +13,19 @@ func (err ResourceNotFoundErrorWrapper) Error() string { func (err ResourceNotFoundErrorWrapper) Unwrap() error { return err.OriginalError } + +type UserMismatchError struct { + chatSessionID string + userID string +} + +func NewUserMismatchError(chatSessionID, userID string) *UserMismatchError { + return &UserMismatchError{ + chatSessionID: chatSessionID, + userID: userID, + } +} + +func (err UserMismatchError) Error() string { + return "chatSession " + err.chatSessionID + " does not belong to user " + err.userID +} diff --git a/pkg/server/routes.go b/pkg/server/routes.go index 7ccb740..765f779 100644 --- a/pkg/server/routes.go +++ b/pkg/server/routes.go @@ -49,12 +49,16 @@ func (s *Server) initializeRoutes() { chatSessionRepository := repositories.NewChatSessionRepository(s.DB) chatSessionService := services.NewChatSessionService(s.logger, chatSessionRepository) + messageRepository := repositories.NewMessageRepository(s.DB) + messageService := services.NewMessageService(s.logger, messageRepository, chatSessionRepository) createChatSessionHandler := chatSessions.NewCreateUserChatSessionHandler(chatSessionService, s.logger) getChatSessionHandler := chatSessions.NewGetChatSessionHandler(chatSessionService, s.logger) + sendMessageHandler := chatSessions.NewSendMessageHandler(messageService, s.logger) - protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionAssetController).Methods("POST") + protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionController).Methods("POST") protected.HandleFunc("/users/{user_id}/chat-sessions", getChatSessionHandler.GetUserChatSessionsController).Methods("GET") + protected.HandleFunc("/users/{user_id}/chat-sessions/{session_id}/messages", sendMessageHandler.SendMessageController).Methods("POST") protected.HandleFunc("/chat-sessions/{session_id}", getChatSessionHandler.GetChatSessionController).Methods("GET") From c6d139e8775d02cbec682ec1ec0b10abac65336e Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 14:06:59 +0300 Subject: [PATCH 10/52] Fixes swagger definitions --- docs/swagger/docs.go | 203 +++++++++++++++++- docs/swagger/swagger.json | 203 +++++++++++++++++- docs/swagger/swagger.yaml | 133 +++++++++++- .../chatSessions/getChatSessionHandler.go | 18 ++ .../chatSessions/sendMessageHandler.go | 12 +- 5 files changed, 549 insertions(+), 20 deletions(-) diff --git a/docs/swagger/docs.go b/docs/swagger/docs.go index 2c41e12..fbecc66 100644 --- a/docs/swagger/docs.go +++ b/docs/swagger/docs.go @@ -24,7 +24,97 @@ const docTemplate = `{ "host": "{{.Host}}", "basePath": "{{.BasePath}}", "paths": { + "/chat-sessions/session_id": { + "get": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Gets chat session", + "summary": "Gets chat session", + "parameters": [ + { + "type": "integer", + "description": "session id", + "name": "session_id", + "in": "path", + "required": true + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + } + } + } + }, "/users/user_id/chat-sessions": { + "get": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Gets all User's chat sessions", + "summary": "Gets all User's chat sessions", + "parameters": [ + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + } + } + }, "post": { "security": [ { @@ -69,6 +159,70 @@ const docTemplate = `{ } } } + }, + "/users/user_id/chat-sessions/session_id/messages": { + "post": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Sends message to a given chat session and gets response", + "summary": "Sends message to a given chat session and gets response", + "parameters": [ + { + "description": "request body", + "name": "SendMessageRequest", + "in": "body", + "required": true, + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageRequest" + } + }, + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + }, + { + "description": "session id", + "name": "session_id", + "in": "body", + "required": true, + "schema": { + "type": "integer" + } + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + } + } + } } }, "definitions": { @@ -87,7 +241,7 @@ const docTemplate = `{ "messages": { "type": "array", "items": { - "$ref": "#/definitions/chatSessions.Message" + "$ref": "#/definitions/chatSessions.MessageResponse" } }, "title": { @@ -98,21 +252,60 @@ const docTemplate = `{ } } }, - "chatSessions.Message": { + "chatSessions.MessageResponse": { "type": "object", "properties": { "content": { "type": "string" }, + "created_at": { + "type": "string" + }, + "errorMessage": { + "type": "string" + }, "id": { "type": "string" }, "sender": { - "type": "string", - "example": "USER" + "type": "string" + } + } + }, + "chatSessions.SendMessageRequest": { + "type": "object", + "properties": { + "content": { + "description": "UserID string ` + "`" + `json:\"chatSessionID\"` + "`" + `\nSessionID string ` + "`" + `json:\"sessionId\"` + "`" + `", + "type": "string" + } + } + }, + "chatSessions.SendMessageResponse": { + "type": "object", + "properties": { + "errorMessage": { + "type": "string" }, - "timestamp": { + "systemMessage": { + "$ref": "#/definitions/chatSessions.MessageResponse" + }, + "userMessage": { + "$ref": "#/definitions/chatSessions.MessageResponse" + } + } + }, + "chatSessions.UserChatSessionsResponse": { + "type": "object", + "properties": { + "errorMessage": { "type": "string" + }, + "sessions": { + "type": "array", + "items": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } } } } diff --git a/docs/swagger/swagger.json b/docs/swagger/swagger.json index 9e27086..4d844be 100644 --- a/docs/swagger/swagger.json +++ b/docs/swagger/swagger.json @@ -18,7 +18,97 @@ "host": "localhost:8080", "basePath": "/", "paths": { + "/chat-sessions/session_id": { + "get": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Gets chat session", + "summary": "Gets chat session", + "parameters": [ + { + "type": "integer", + "description": "session id", + "name": "session_id", + "in": "path", + "required": true + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } + } + } + } + }, "/users/user_id/chat-sessions": { + "get": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Gets all User's chat sessions", + "summary": "Gets all User's chat sessions", + "parameters": [ + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.UserChatSessionsResponse" + } + } + } + }, "post": { "security": [ { @@ -63,6 +153,70 @@ } } } + }, + "/users/user_id/chat-sessions/session_id/messages": { + "post": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Sends message to a given chat session and gets response", + "summary": "Sends message to a given chat session and gets response", + "parameters": [ + { + "description": "request body", + "name": "SendMessageRequest", + "in": "body", + "required": true, + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageRequest" + } + }, + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + }, + { + "description": "session id", + "name": "session_id", + "in": "body", + "required": true, + "schema": { + "type": "integer" + } + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + } + } + } } }, "definitions": { @@ -81,7 +235,7 @@ "messages": { "type": "array", "items": { - "$ref": "#/definitions/chatSessions.Message" + "$ref": "#/definitions/chatSessions.MessageResponse" } }, "title": { @@ -92,21 +246,60 @@ } } }, - "chatSessions.Message": { + "chatSessions.MessageResponse": { "type": "object", "properties": { "content": { "type": "string" }, + "created_at": { + "type": "string" + }, + "errorMessage": { + "type": "string" + }, "id": { "type": "string" }, "sender": { - "type": "string", - "example": "USER" + "type": "string" + } + } + }, + "chatSessions.SendMessageRequest": { + "type": "object", + "properties": { + "content": { + "description": "UserID string `json:\"chatSessionID\"`\nSessionID string `json:\"sessionId\"`", + "type": "string" + } + } + }, + "chatSessions.SendMessageResponse": { + "type": "object", + "properties": { + "errorMessage": { + "type": "string" }, - "timestamp": { + "systemMessage": { + "$ref": "#/definitions/chatSessions.MessageResponse" + }, + "userMessage": { + "$ref": "#/definitions/chatSessions.MessageResponse" + } + } + }, + "chatSessions.UserChatSessionsResponse": { + "type": "object", + "properties": { + "errorMessage": { "type": "string" + }, + "sessions": { + "type": "array", + "items": { + "$ref": "#/definitions/chatSessions.ChatSessionResponse" + } } } } diff --git a/docs/swagger/swagger.yaml b/docs/swagger/swagger.yaml index 03fd6fb..ef446fb 100644 --- a/docs/swagger/swagger.yaml +++ b/docs/swagger/swagger.yaml @@ -12,25 +12,52 @@ definitions: type: string messages: items: - $ref: '#/definitions/chatSessions.Message' + $ref: '#/definitions/chatSessions.MessageResponse' type: array title: type: string updatedAt: type: string type: object - chatSessions.Message: + chatSessions.MessageResponse: properties: content: type: string + created_at: + type: string + errorMessage: + type: string id: type: string sender: - example: USER type: string - timestamp: + type: object + chatSessions.SendMessageRequest: + properties: + content: + description: |- + UserID string `json:"chatSessionID"` + SessionID string `json:"sessionId"` type: string type: object + chatSessions.SendMessageResponse: + properties: + errorMessage: + type: string + systemMessage: + $ref: '#/definitions/chatSessions.MessageResponse' + userMessage: + $ref: '#/definitions/chatSessions.MessageResponse' + type: object + chatSessions.UserChatSessionsResponse: + properties: + errorMessage: + type: string + sessions: + items: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + type: array + type: object host: localhost:8080 info: contact: @@ -40,7 +67,64 @@ info: title: Louk Chatwalker version: "1.0" paths: + /chat-sessions/session_id: + get: + description: Gets chat session + parameters: + - description: session id + in: path + name: session_id + required: true + type: integer + responses: + "201": + description: Created + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + "400": + description: Error in message payload + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + "401": + description: Authentication error + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + "500": + description: Internal Server Error + schema: + $ref: '#/definitions/chatSessions.ChatSessionResponse' + security: + - BearerAuth: [] + summary: Gets chat session /users/user_id/chat-sessions: + get: + description: Gets all User's chat sessions + parameters: + - description: user id + in: path + name: user_id + required: true + type: integer + responses: + "201": + description: Created + schema: + $ref: '#/definitions/chatSessions.UserChatSessionsResponse' + "400": + description: Error in message payload + schema: + $ref: '#/definitions/chatSessions.UserChatSessionsResponse' + "401": + description: Authentication error + schema: + $ref: '#/definitions/chatSessions.UserChatSessionsResponse' + "500": + description: Internal Server Error + schema: + $ref: '#/definitions/chatSessions.UserChatSessionsResponse' + security: + - BearerAuth: [] + summary: Gets all User's chat sessions post: description: Creates a chat session for User parameters: @@ -69,6 +153,47 @@ paths: security: - BearerAuth: [] summary: Creates chat session + /users/user_id/chat-sessions/session_id/messages: + post: + description: Sends message to a given chat session and gets response + parameters: + - description: request body + in: body + name: SendMessageRequest + required: true + schema: + $ref: '#/definitions/chatSessions.SendMessageRequest' + - description: user id + in: path + name: user_id + required: true + type: integer + - description: session id + in: body + name: session_id + required: true + schema: + type: integer + responses: + "201": + description: Created + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + "400": + description: Error in message payload + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + "401": + description: Authentication error + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + "500": + description: Internal Server Error + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + security: + - BearerAuth: [] + summary: Sends message to a given chat session and gets response produces: - application/json securityDefinitions: diff --git a/internal/handlers/chatSessions/getChatSessionHandler.go b/internal/handlers/chatSessions/getChatSessionHandler.go index aa053a8..b4fd577 100644 --- a/internal/handlers/chatSessions/getChatSessionHandler.go +++ b/internal/handlers/chatSessions/getChatSessionHandler.go @@ -25,6 +25,15 @@ func NewGetChatSessionHandler( } } +// @Summary Gets all User's chat sessions +// @Description Gets all User's chat sessions +// @Security BearerAuth +// @Param user_id path int true "user id" +// @Success 201 {object} UserChatSessionsResponse +// @Failure 400 {object} UserChatSessionsResponse "Error in message payload" +// @Failure 401 {object} UserChatSessionsResponse "Authentication error" +// @Failure 500 {object} UserChatSessionsResponse "Internal Server Error" +// @Router /users/user_id/chat-sessions [get] func (handler *GetChatSessionHandler) GetUserChatSessionsController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") @@ -85,6 +94,15 @@ func (handler *GetChatSessionHandler) GetUserChatSessionsController(w http.Respo handler.JsonUserChatSessionResponse(w, http.StatusOK, response) } +// @Summary Gets chat session +// @Description Gets chat session +// @Security BearerAuth +// @Param session_id path int true "session id" +// @Success 201 {object} ChatSessionResponse +// @Failure 400 {object} ChatSessionResponse "Error in message payload" +// @Failure 401 {object} ChatSessionResponse "Authentication error" +// @Failure 500 {object} ChatSessionResponse "Internal Server Error" +// @Router /chat-sessions/session_id [get] func (handler *GetChatSessionHandler) GetChatSessionController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") diff --git a/internal/handlers/chatSessions/sendMessageHandler.go b/internal/handlers/chatSessions/sendMessageHandler.go index e03b1e9..f89e1c5 100644 --- a/internal/handlers/chatSessions/sendMessageHandler.go +++ b/internal/handlers/chatSessions/sendMessageHandler.go @@ -30,12 +30,12 @@ func NewSendMessageHandler( // @Description Sends message to a given chat session and gets response // @Security BearerAuth // @Param SendMessageRequest body SendMessageRequest true "request body" -// @Param user_id path int true "user id" -// @Param session_id body int true "session id" -// @Success 201 {object} MessageResponse -// @Failure 400 {object} MessageResponse "Error in message payload" -// @Failure 401 {object} MessageResponse "Authentication error" -// @Failure 500 {object} MessageResponse "Internal Server Error" +// @Param user_id path int true "user id" +// @Param session_id body int true "session id" +// @Success 201 {object} SendMessageResponse +// @Failure 400 {object} SendMessageResponse "Error in message payload" +// @Failure 401 {object} SendMessageResponse "Authentication error" +// @Failure 500 {object} SendMessageResponse "Internal Server Error" // @Router /users/user_id/chat-sessions/session_id/messages [post] func (handler *SendMessageHandler) SendMessageController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") From 6e3bd4cbec6430d0ce7d3d8ceb2410f72ae39d0f Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 14:50:25 +0300 Subject: [PATCH 11/52] Sets answer from LLM --- cmd/http/main.go | 94 +++++++++++++----------- internal/core/services/messageService.go | 64 +++++++++++++++- pkg/server/routes.go | 2 +- pkg/server/server.go | 29 +++++--- pkg/vectordb/pinecone.go | 39 ++++++++-- 5 files changed, 170 insertions(+), 58 deletions(-) diff --git a/cmd/http/main.go b/cmd/http/main.go index 902330a..420a8e5 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -11,8 +11,10 @@ import ( "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "github.com/loukaspe/jedi-team-challenge/pkg/server" + "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" "github.com/openai/openai-go" "github.com/openai/openai-go/option" + "github.com/pinecone-io/go-pinecone/v3/pinecone" "github.com/pkoukk/tiktoken-go" log "github.com/sirupsen/logrus" "gorm.io/driver/postgres" @@ -29,8 +31,9 @@ func main() { client := getOpenAIClient() chunker := getChunker(encoder) embedder := getEmbedder(&client) + pineconeVectorDB := getPineconeVectorDB() - inputKnowledgeBase(chunker, embedder) + inputKnowledgeBase(chunker, embedder, pineconeVectorDB) logger := logger.NewLogger(context.Background()) router := mux.NewRouter() @@ -40,7 +43,7 @@ func main() { } db := getDB() - server := server.NewServer(db, router, httpServer, logger) + server := server.NewServer(db, router, httpServer, logger, &client, embedder, pineconeVectorDB) server.Run() } @@ -141,43 +144,52 @@ func getOpenAIClient() openai.Client { return openai.NewClient(option.WithAPIKey(os.Getenv("OPENAI_API_KEY"))) } -func inputKnowledgeBase(chunker *chunks.Chunker, embedder *embeddings.EmbeddingService) { - - //textBytes, err := os.ReadFile("./data.md") - //if err != nil { - // log.Fatal(err) - //} - //text := string(textBytes) - // - //chunks := chunker.Chunk(text) - //fmt.Printf("Generated %d chunks\n", len(chunks)) - // - //ctx := context.Background() - //embeddings, err := embedder.Embed(ctx, chunks) - //if err != nil { - // log.Fatalf("Embedding error: %v", err) - //} - // - //// Output - //for i, emb := range embeddings { - // fmt.Printf("Chunk %d → %d-dim vector\n", i, len(emb)) - //} - // - //pineconeClient, err := pinecone.NewClient(pinecone.NewClientParams{ - // ApiKey: os.Getenv("PINECONE_API_KEY"), - //}) - //if err != nil { - // log.Fatalf("Failed to create pinecone Client: %v", err) - //} - //pineconeVectorDB := vectordb.NewPineconeVectorDB( - // os.Getenv("PINECONE_INDEX"), - // pineconeClient, - //) - // - //count, err := pineconeVectorDB.StoreEmbeddings(ctx, embeddings) - //if err != nil { - // log.Fatalf("Failed to store embeddings: %v", err) - //} - // - //fmt.Sprintf("Stored %d embeddings in Pinecone index %s\n", count, os.Getenv("PINECONE_INDEX")) +func getPineconeVectorDB() *vectordb.PineconeVectorDB { + topKResultsNumberAsString := os.Getenv("TOP_K_RESULTS_NUMBER") + topKResultsNumber, err := strconv.Atoi(topKResultsNumberAsString) + if err != nil { + log.Fatal("Cannot read top k results number: ", err) + } + + pineconeClient, err := pinecone.NewClient(pinecone.NewClientParams{ + ApiKey: os.Getenv("PINECONE_API_KEY"), + }) + if err != nil { + log.Fatalf("Failed to create pinecone Client: %v", err) + } + return vectordb.NewPineconeVectorDB( + topKResultsNumber, + os.Getenv("PINECONE_INDEX"), + pineconeClient, + ) + +} + +func inputKnowledgeBase(chunker *chunks.Chunker, embedder *embeddings.EmbeddingService, pineconeVectorDB *vectordb.PineconeVectorDB) { + textBytes, err := os.ReadFile("./data.md") + if err != nil { + log.Fatal(err) + } + text := string(textBytes) + + chunks := chunker.Chunk(text) + fmt.Printf("Generated %d chunks\n", len(chunks)) + + ctx := context.Background() + embeddings, err := embedder.Embed(ctx, chunks) + if err != nil { + log.Fatalf("Embedding error: %v", err) + } + + // Output + for i, emb := range embeddings { + fmt.Printf("Chunk %d → %d-dim vector\n", i, len(emb)) + } + + count, err := pineconeVectorDB.StoreEmbeddings(ctx, embeddings) + if err != nil { + log.Fatalf("Failed to store embeddings: %v", err) + } + + fmt.Sprintf("Stored %d embeddings in Pinecone index %s\n", count, os.Getenv("PINECONE_INDEX")) } diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index 01e7151..e3951b9 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -2,11 +2,15 @@ package services import ( "context" + "errors" + "fmt" "github.com/google/uuid" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/core/ports" apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/openai/openai-go" + "strings" ) type MessageServiceInterface interface { @@ -14,21 +18,38 @@ type MessageServiceInterface interface { GetAnswerForMessage(context.Context, uuid.UUID) (*domain.Message, error) } +type Embedder interface { + Embed(ctx context.Context, inputs []string) ([][]float64, error) +} + +type VectorDB interface { + SemanticSearch(ctx context.Context, embeddings []float32) ([]string, error) +} + type MessageService struct { logger logger.LoggerInterface messageRepository ports.MessageRepositoryInterface chatSessionRepository ports.ChatSessionRepositoryInterface + embedder Embedder + vectorDB VectorDB + openAIClient *openai.Client } func NewMessageService( logger logger.LoggerInterface, messageRepositoryInterface ports.MessageRepositoryInterface, chatSessionRepository ports.ChatSessionRepositoryInterface, + embedder Embedder, + vectorDB VectorDB, + openAIClient *openai.Client, ) *MessageService { return &MessageService{ logger: logger, messageRepository: messageRepositoryInterface, chatSessionRepository: chatSessionRepository, + embedder: embedder, + vectorDB: vectorDB, + openAIClient: openAIClient, } } @@ -51,14 +72,53 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI return nil, err } + embeddings, err := s.embedder.Embed(ctx, []string{ + initialMessage.Content, + }) + if err != nil { + return nil, err + } + + // we only have on text so we only care for the first embedding row + vectorToFloat32 := make([]float32, len(embeddings[0])) + for i, v := range embeddings[0] { + vectorToFloat32[i] = float32(v) + } + + accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(ctx, vectorToFloat32) + + prompt := fmt.Sprintf(`Use the following context to answer the question. + Context: + %s + + Question: + %s + + Answer:`, + strings.Join(accumulatedTextFromSearch, "\n"), + initialMessage.Content, + ) + + chatCompletion, err := s.openAIClient.Chat.Completions.New(ctx, openai.ChatCompletionNewParams{ + Messages: []openai.ChatCompletionMessageParamUnion{ + openai.UserMessage(prompt), + }, + }) + + answer := chatCompletion.Choices[0].Message.Content + + if answer == "" { + // TODO: mh ta les apierrors + return nil, errors.New("received empty response from LLM") + } + replyMessage := &domain.Message{ ID: initialMessageID, ChatSessionID: initialMessage.ChatSessionID, - Content: "This is a mock response from the LLM", + Content: answer, Sender: "SYSTEM", //TODO: constant } - //TODO: get from llm insertedMessageID, err := s.messageRepository.CreateMessage(ctx, replyMessage) if err != nil { return nil, err diff --git a/pkg/server/routes.go b/pkg/server/routes.go index 765f779..a6baff0 100644 --- a/pkg/server/routes.go +++ b/pkg/server/routes.go @@ -50,7 +50,7 @@ func (s *Server) initializeRoutes() { chatSessionRepository := repositories.NewChatSessionRepository(s.DB) chatSessionService := services.NewChatSessionService(s.logger, chatSessionRepository) messageRepository := repositories.NewMessageRepository(s.DB) - messageService := services.NewMessageService(s.logger, messageRepository, chatSessionRepository) + messageService := services.NewMessageService(s.logger, messageRepository, chatSessionRepository, s.embedder, s.pineconeVectorDB, s.openAIClient) createChatSessionHandler := chatSessions.NewCreateUserChatSessionHandler(chatSessionService, s.logger) getChatSessionHandler := chatSessions.NewGetChatSessionHandler(chatSessionService, s.logger) diff --git a/pkg/server/server.go b/pkg/server/server.go index 9a715cc..0329424 100644 --- a/pkg/server/server.go +++ b/pkg/server/server.go @@ -4,7 +4,10 @@ import ( "context" "errors" "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" + "github.com/openai/openai-go" log "github.com/sirupsen/logrus" "gorm.io/gorm" "net/http" @@ -15,10 +18,13 @@ import ( ) type Server struct { - DB *gorm.DB - httpServer *http.Server - router *mux.Router - logger logger.LoggerInterface + DB *gorm.DB + httpServer *http.Server + router *mux.Router + logger logger.LoggerInterface + openAIClient *openai.Client + embedder *embeddings.EmbeddingService + pineconeVectorDB *vectordb.PineconeVectorDB } func NewServer( @@ -26,13 +32,18 @@ func NewServer( router *mux.Router, httpServer *http.Server, logger logger.LoggerInterface, - + openAIClient *openai.Client, + embedder *embeddings.EmbeddingService, + pineconeVectorDB *vectordb.PineconeVectorDB, ) *Server { return &Server{ - DB: db, - router: router, - httpServer: httpServer, - logger: logger, + DB: db, + router: router, + httpServer: httpServer, + logger: logger, + openAIClient: openAIClient, + embedder: embedder, + pineconeVectorDB: pineconeVectorDB, } } diff --git a/pkg/vectordb/pinecone.go b/pkg/vectordb/pinecone.go index 5b61fc3..36234dc 100644 --- a/pkg/vectordb/pinecone.go +++ b/pkg/vectordb/pinecone.go @@ -7,14 +7,16 @@ import ( ) type PineconeVectorDB struct { - client *pinecone.Client - index string + client *pinecone.Client + index string + topKResultsNumber int } -func NewPineconeVectorDB(index string, client *pinecone.Client) *PineconeVectorDB { +func NewPineconeVectorDB(topKResultsNumber int, index string, client *pinecone.Client) *PineconeVectorDB { return &PineconeVectorDB{ - index: index, - client: client, + topKResultsNumber: topKResultsNumber, + index: index, + client: client, } } @@ -61,3 +63,30 @@ func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings [][] return int(count), nil } + +func (db *PineconeVectorDB) SemanticSearch(ctx context.Context, embeddings []float32) ([]string, error) { + idx, err := db.client.DescribeIndex(ctx, db.index) + if err != nil { + return []string{}, err + } + + idxConnection, err := db.client.Index(pinecone.NewIndexConnParams{Host: idx.Host}) + if err != nil { + return []string{}, err + } + + res, err := idxConnection.QueryByVectorValues(ctx, &pinecone.QueryByVectorValuesRequest{ + Vector: embeddings, + TopK: uint32(db.topKResultsNumber), + IncludeValues: false, + IncludeMetadata: true, + }) + + var contextTexts []string + for _, match := range res.Matches { + text := match.Vector.Metadata.String() + contextTexts = append(contextTexts, text) + } + + return contextTexts, nil +} From 3e38a6f18ec79ad2df7ce9ab5d106f0297812461 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 15:22:43 +0300 Subject: [PATCH 12/52] Inserts LLM into answer generation --- cmd/http/main.go | 11 ++++++++- internal/core/domain/embeddings.go | 6 +++++ internal/core/services/messageService.go | 12 ++++++---- pkg/vectordb/pinecone.go | 30 ++++++++++++------------ 4 files changed, 39 insertions(+), 20 deletions(-) create mode 100644 internal/core/domain/embeddings.go diff --git a/cmd/http/main.go b/cmd/http/main.go index 420a8e5..a25e6d2 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -6,6 +6,7 @@ import ( "github.com/google/uuid" "github.com/gorilla/mux" "github.com/joho/godotenv" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/repositories" "github.com/loukaspe/jedi-team-challenge/pkg/chunks" "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" @@ -186,7 +187,15 @@ func inputKnowledgeBase(chunker *chunks.Chunker, embedder *embeddings.EmbeddingS fmt.Printf("Chunk %d → %d-dim vector\n", i, len(emb)) } - count, err := pineconeVectorDB.StoreEmbeddings(ctx, embeddings) + domainEmbeddings := make([]*domain.Embeddings, 0, len(embeddings)) + for i, e := range embeddings { + domainEmbeddings = append(domainEmbeddings, &domain.Embeddings{ + Embeddings: e, + Text: chunks[i], + }) + } + + count, err := pineconeVectorDB.StoreEmbeddings(ctx, domainEmbeddings) if err != nil { log.Fatalf("Failed to store embeddings: %v", err) } diff --git a/internal/core/domain/embeddings.go b/internal/core/domain/embeddings.go new file mode 100644 index 0000000..920f608 --- /dev/null +++ b/internal/core/domain/embeddings.go @@ -0,0 +1,6 @@ +package domain + +type Embeddings struct { + Embeddings []float64 `json:"embedding"` + Text string `json:"text"` +} diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index e3951b9..9bee4a0 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -72,8 +72,8 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI return nil, err } - embeddings, err := s.embedder.Embed(ctx, []string{ - initialMessage.Content, + embeddings, err := s.embedder.Embed(context.Background(), []string{ + "What do you know about Gen Z in Nashville", }) if err != nil { return nil, err @@ -85,7 +85,7 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI vectorToFloat32[i] = float32(v) } - accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(ctx, vectorToFloat32) + accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(context.Background(), vectorToFloat32) prompt := fmt.Sprintf(`Use the following context to answer the question. Context: @@ -99,11 +99,15 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI initialMessage.Content, ) - chatCompletion, err := s.openAIClient.Chat.Completions.New(ctx, openai.ChatCompletionNewParams{ + chatCompletion, err := s.openAIClient.Chat.Completions.New(context.Background(), openai.ChatCompletionNewParams{ Messages: []openai.ChatCompletionMessageParamUnion{ openai.UserMessage(prompt), }, + Model: openai.ChatModelGPT4o, }) + if err != nil { + return nil, err + } answer := chatCompletion.Choices[0].Message.Content diff --git a/pkg/vectordb/pinecone.go b/pkg/vectordb/pinecone.go index 36234dc..55209f5 100644 --- a/pkg/vectordb/pinecone.go +++ b/pkg/vectordb/pinecone.go @@ -3,7 +3,9 @@ package vectordb import ( "context" "fmt" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/pinecone-io/go-pinecone/v3/pinecone" + "google.golang.org/protobuf/types/known/structpb" ) type PineconeVectorDB struct { @@ -20,7 +22,7 @@ func NewPineconeVectorDB(topKResultsNumber int, index string, client *pinecone.C } } -func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings [][]float64) (int, error) { +func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings []*domain.Embeddings) (int, error) { idx, err := db.client.DescribeIndex(ctx, db.index) if err != nil { return 0, err @@ -32,27 +34,25 @@ func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings [][] } vectors := make([]*pinecone.Vector, len(embeddings)) - for i, vec := range embeddings { + for i, embedding := range embeddings { id := fmt.Sprintf("doc1-chunk-%d", i) - //md, err := structpb.NewStruct(map[string]interface{}{ - // "text": chunks[i], - //}) - //if err != nil { - // log.Fatalf("failed to create metadata struct: %v", err) - //} + md, err := structpb.NewStruct(map[string]interface{}{ + "text": embeddings[i].Text, + }) + if err != nil { + return 0, err + } - vectorToFloat32 := make([]float32, len(vec)) - for i, v := range vec { + vectorToFloat32 := make([]float32, len(embedding.Embeddings)) + for i, v := range embeddings[i].Embeddings { vectorToFloat32[i] = float32(v) } vectors[i] = &pinecone.Vector{ - Id: id, - Values: &vectorToFloat32, - //Metadata: map[string]interface{}{ - // "text": chunks[i], // store the original chunk if you want - //}, + Id: id, + Values: &vectorToFloat32, + Metadata: md, } } From 1483db9fe00f5472fba16a4867c963be840a2f10 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 15:29:58 +0300 Subject: [PATCH 13/52] Adds curl examples --- examples/createChatSessionForUser.http | 7 +++++++ examples/getChatSessionByID.http | 7 +++++++ examples/getChatSessionByUser.http | 7 +++++++ examples/sendMessage.http | 16 ++++++++++++++++ examples/token.http | 11 +++++++++-- 5 files changed, 46 insertions(+), 2 deletions(-) create mode 100644 examples/createChatSessionForUser.http create mode 100644 examples/getChatSessionByID.http create mode 100644 examples/getChatSessionByUser.http create mode 100644 examples/sendMessage.http diff --git a/examples/createChatSessionForUser.http b/examples/createChatSessionForUser.http new file mode 100644 index 0000000..9074df9 --- /dev/null +++ b/examples/createChatSessionForUser.http @@ -0,0 +1,7 @@ +# curl --location --request POST 'localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions' +#--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8' +POST localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions +Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8 + +### + diff --git a/examples/getChatSessionByID.http b/examples/getChatSessionByID.http new file mode 100644 index 0000000..fa10808 --- /dev/null +++ b/examples/getChatSessionByID.http @@ -0,0 +1,7 @@ +# curl --location 'localhost:8080/chat-sessions/3ee356a1-640c-44f8-b8d5-a0040385da77' +#--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8' +GET localhost:8080/chat-sessions/3ee356a1-640c-44f8-b8d5-a0040385da77 +Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8 + +### + diff --git a/examples/getChatSessionByUser.http b/examples/getChatSessionByUser.http new file mode 100644 index 0000000..3e284a2 --- /dev/null +++ b/examples/getChatSessionByUser.http @@ -0,0 +1,7 @@ +# curl --location 'localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions' +#--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8' +GET localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions +Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8 + +### + diff --git a/examples/sendMessage.http b/examples/sendMessage.http new file mode 100644 index 0000000..6e299af --- /dev/null +++ b/examples/sendMessage.http @@ -0,0 +1,16 @@ +# curl --location 'localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions/367c5892-ed03-4d5f-a0d6-907b791d7a2f/messages' +#--header 'Content-Type: application/json' +#--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8' +#--data '{ +# "content":"what do you know about gen z in nashville" +#}' +POST localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions/367c5892-ed03-4d5f-a0d6-907b791d7a2f/messages +Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4MzUxODc0LCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.U_aeQ2PCjSnUCnOUy80k8QELoOIIquLOUVpm_6hmVs8 +Content-Type: application/json + +{ + "content": "what do you know about gen z in nashville" +} + +### + diff --git a/examples/token.http b/examples/token.http index dab3c17..8df1a59 100644 --- a/examples/token.http +++ b/examples/token.http @@ -1,9 +1,16 @@ -POST http://localhost:8080/token +# curl --location 'localhost:8080/token' +#--header 'Content-Type: application/json' +#--data '{ +# "username": "user", +# "password": "password" +#}' +POST localhost:8080/token Content-Type: application/json { - "username": "username", + "username": "user", "password": "password" } ### + From cb1a530479e62afcfe71170efd663a44a9455292 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 16:33:29 +0300 Subject: [PATCH 14/52] Adds an e2e script for testing all functionalities --- script/e2e.sh | 74 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 74 insertions(+) create mode 100644 script/e2e.sh diff --git a/script/e2e.sh b/script/e2e.sh new file mode 100644 index 0000000..3dea0c6 --- /dev/null +++ b/script/e2e.sh @@ -0,0 +1,74 @@ +#!/bin/bash + +# Define the backend API URL +BASE_URL="http://localhost:8080" +TOKEN_ENDPOINT="/token" +USER_ID="12345678-0000-0000-0000-000000000000" + +# Step 1: Get the token +response_token=$(curl -s --location "$BASE_URL$TOKEN_ENDPOINT" \ + --header 'Content-Type: application/json' \ + --data '{"username": "user", "password": "password"}') + +# Extract the token from the response using jq (ensure jq is installed) +token=$(echo "$response_token" | jq -r '.token') + +# Check if the token was successfully retrieved +if [ -z "$token" ]; then + echo "Error: Failed to retrieve token." + exit 1 +fi + +echo "Token retrieved successfully: $token" +echo "----------------------------------" + +# Step 2: Create 3 chat sessions for the user and get the first session's ID +for i in {1..3}; do + # Make a POST request to create a chat session + response=$(curl -s --location --request POST "$BASE_URL/users/$USER_ID/chat-sessions" \ + --header "Authorization: Bearer $token") + + # Extract the chat session ID from the response of the first chat session + if [ $i -eq 1 ]; then + CHAT_SESSION_ID=$(echo "$response" | jq -r '.id') + echo "First chat session created with ID: $CHAT_SESSION_ID" + fi + + # Check if the session creation was successful +# if [[ $(echo "$response" | jq -r '.status') == "success" ]]; then +# echo "Chat session $i created successfully." +# else +# echo "Error creating chat session $i." +# exit +# fi +done + +# Step 3: Send the first message +message_content_1="what do you know about gen z in nashville" +response_message_1=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ + --header "Content-Type: application/json" \ + --header "Authorization: Bearer $token" \ + --data "{\"content\":\"$message_content_1\"}") + +# Step 4: Send the second message +message_content_2="what do you know about latino mobile gamers" +response_message_2=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ + --header "Content-Type: application/json" \ + --header "Authorization: Bearer $token" \ + --data "{\"content\":\"$message_content_2\"}") + +# Step 5: Display the responses from sending the messages +echo "Message 1 sent to chat session $CHAT_SESSION_ID:" +echo "$response_message_1" + +echo "Message 2 sent to chat session $CHAT_SESSION_ID:" +echo "$response_message_2" + +# Step 6: Fetch and display the whole chat session +echo "Fetching the entire chat session $CHAT_SESSION_ID..." +chat_session_response=$(curl -s --location "$BASE_URL/chat-sessions/$CHAT_SESSION_ID" \ + --header "Authorization: Bearer $token") + +# Display the full chat session +echo "Full Chat Session $CHAT_SESSION_ID:" +echo "$chat_session_response" From a27356856f121ca601616ecfc983ebc4eb0ae449 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 17:48:36 +0300 Subject: [PATCH 15/52] Small fixes --- internal/handlers/chatSessions/sendMessageHandler.go | 3 ++- internal/repositories/chatSessionDao.go | 3 +++ pkg/embeddings/embedder.go | 5 ----- pkg/errors/errors.go | 2 +- pkg/helpers/vectors.go | 9 +++++++++ pkg/vectordb/pinecone.go | 6 ++---- 6 files changed, 17 insertions(+), 11 deletions(-) create mode 100644 pkg/helpers/vectors.go diff --git a/internal/handlers/chatSessions/sendMessageHandler.go b/internal/handlers/chatSessions/sendMessageHandler.go index f89e1c5..a36e916 100644 --- a/internal/handlers/chatSessions/sendMessageHandler.go +++ b/internal/handlers/chatSessions/sendMessageHandler.go @@ -6,6 +6,7 @@ import ( "github.com/gorilla/mux" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/core/services" + "github.com/loukaspe/jedi-team-challenge/internal/repositories" apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "net/http" @@ -109,7 +110,7 @@ func (handler *SendMessageHandler) SendMessageController(w http.ResponseWriter, domainMessage := &domain.Message{ ChatSessionID: chatSessionID, Content: request.Content, - Sender: "USER", //TODO: hardcode for now, should be set by the user + Sender: repositories.USER_SENDER, } insertedUUID, err := handler.MessageService.CreateMessage( diff --git a/internal/repositories/chatSessionDao.go b/internal/repositories/chatSessionDao.go index cd6da33..462c1d6 100644 --- a/internal/repositories/chatSessionDao.go +++ b/internal/repositories/chatSessionDao.go @@ -5,6 +5,9 @@ import ( "time" ) +const SYSTEM_SENDER = "SYSTEM" +const USER_SENDER = "USER" + type ChatSession struct { ID uuid.UUID `gorm:"type:uuid;primaryKey;default:uuid_generate_v4()"` UserID uuid.UUID `gorm:"type:uuid;not null;index"` diff --git a/pkg/embeddings/embedder.go b/pkg/embeddings/embedder.go index 0698104..1b55422 100644 --- a/pkg/embeddings/embedder.go +++ b/pkg/embeddings/embedder.go @@ -11,12 +11,7 @@ import ( const maxEmbeddingRetries = 5 -//type Chunker interface { -// Chunk(text string) []string -//} - type EmbeddingService struct { - // TODO: exei nohma edw to interface OpenAIClient *openai.Client EmbeddingModel openai.EmbeddingModel } diff --git a/pkg/errors/errors.go b/pkg/errors/errors.go index 69a0943..13ac904 100644 --- a/pkg/errors/errors.go +++ b/pkg/errors/errors.go @@ -1,4 +1,4 @@ -package apierrors +package customerrors type ResourceNotFoundErrorWrapper struct { OriginalError error diff --git a/pkg/helpers/vectors.go b/pkg/helpers/vectors.go new file mode 100644 index 0000000..9188eba --- /dev/null +++ b/pkg/helpers/vectors.go @@ -0,0 +1,9 @@ +package helpers + +func Float64ToFloat32(input []float64) []float32 { + output := make([]float32, len(input)) + for i, v := range input { + output[i] = float32(v) + } + return output +} diff --git a/pkg/vectordb/pinecone.go b/pkg/vectordb/pinecone.go index 55209f5..6f78e22 100644 --- a/pkg/vectordb/pinecone.go +++ b/pkg/vectordb/pinecone.go @@ -4,6 +4,7 @@ import ( "context" "fmt" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/pkg/helpers" "github.com/pinecone-io/go-pinecone/v3/pinecone" "google.golang.org/protobuf/types/known/structpb" ) @@ -44,10 +45,7 @@ func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings []*d return 0, err } - vectorToFloat32 := make([]float32, len(embedding.Embeddings)) - for i, v := range embeddings[i].Embeddings { - vectorToFloat32[i] = float32(v) - } + vectorToFloat32 := helpers.Float64ToFloat32(embedding.Embeddings) vectors[i] = &pinecone.Vector{ Id: id, From ef85d964a1ab3f4ae4a194bb1947b609503a4ae5 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 17:49:02 +0300 Subject: [PATCH 16/52] Adds update title for chat session --- .../chatSessionRepositoryUpdate.go | 27 ++++ .../chatSessionRepositoryUpdate_test.go | 142 ++++++++++++++++++ 2 files changed, 169 insertions(+) create mode 100644 internal/repositories/chatSessionRepositoryUpdate.go create mode 100644 internal/repositories/chatSessionRepositoryUpdate_test.go diff --git a/internal/repositories/chatSessionRepositoryUpdate.go b/internal/repositories/chatSessionRepositoryUpdate.go new file mode 100644 index 0000000..b43642f --- /dev/null +++ b/internal/repositories/chatSessionRepositoryUpdate.go @@ -0,0 +1,27 @@ +package repositories + +import ( + "context" + "errors" + "github.com/google/uuid" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "gorm.io/gorm" +) + +func (repo *ChatSessionRepository) UpdateChatSessionTitle( + ctx context.Context, + uuid uuid.UUID, + title string, +) error { + err := repo.db.WithContext(ctx).Model(&ChatSession{}). + Where("id = ?", uuid). + Update("title", title).Error + + if err == gorm.ErrRecordNotFound { + return apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("chatSessionID " + uuid.String() + " not found"), + } + } + + return err +} diff --git a/internal/repositories/chatSessionRepositoryUpdate_test.go b/internal/repositories/chatSessionRepositoryUpdate_test.go new file mode 100644 index 0000000..0bc79da --- /dev/null +++ b/internal/repositories/chatSessionRepositoryUpdate_test.go @@ -0,0 +1,142 @@ +package repositories + +import ( + "context" + "errors" + "github.com/DATA-DOG/go-sqlmock" + "github.com/google/uuid" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/stretchr/testify/assert" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "regexp" + "testing" +) + +func TestChatRepository_UpdateChatSessionTitle(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + uuid uuid.UUID + title string + } + tests := []struct { + name string + args args + mockSqlChatQueryExpected string + mockUpdatedChatIdReturned int + }{ + { + name: "valid", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + title: "mockTitle", + }, + mockSqlChatQueryExpected: `UPDATE "chat_sessions" SET "title"=$1,"updated_at"=$2 WHERE id = $3`, + mockUpdatedChatIdReturned: 666, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &ChatSessionRepository{ + db: gormDb, + } + + mockDb.ExpectBegin() + mockDb.ExpectExec(regexp.QuoteMeta(tt.mockSqlChatQueryExpected)). + WithArgs(tt.args.title, sqlmock.AnyArg(), tt.args.uuid). + WillReturnResult(sqlmock.NewResult(int64(tt.mockUpdatedChatIdReturned), 1)) + mockDb.ExpectCommit() + + err := repo.UpdateChatSessionTitle( + context.Background(), + tt.args.uuid, + tt.args.title, + ) + + assert.NoError(t, err) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} + +func TestChatRepository_UpdateChatSessionTitleWithError(t *testing.T) { + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + uuid uuid.UUID + title string + } + tests := []struct { + name string + args args + mockSqlChatQueryExpected string + mockSqlErrorReturned error + expectedError error + }{ + { + name: "random error", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + title: "mockTitle", + }, + mockSqlChatQueryExpected: `UPDATE "chat_sessions" SET "title"=$1,"updated_at"=$2 WHERE id = $3`, + mockSqlErrorReturned: errors.New("random error"), + expectedError: errors.New("random error"), + }, + { + name: "title not found", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + title: "mockTitle", + }, + mockSqlChatQueryExpected: `UPDATE "chat_sessions" SET "title"=$1,"updated_at"=$2 WHERE id = $3`, + mockSqlErrorReturned: gorm.ErrRecordNotFound, + expectedError: apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("chatSessionID 12345678-0000-0000-0000-000000000000 not found"), + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &ChatSessionRepository{ + db: gormDb, + } + + mockDb.ExpectBegin() + mockDb.ExpectExec(regexp.QuoteMeta(tt.mockSqlChatQueryExpected)). + WithArgs(tt.args.title, sqlmock.AnyArg(), tt.args.uuid). + WillReturnError(tt.mockSqlErrorReturned) + mockDb.ExpectRollback() + + actual := repo.UpdateChatSessionTitle( + context.Background(), + tt.args.uuid, + tt.args.title, + ) + + assert.Equal(t, actual, tt.expectedError) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} From a4fab3617a07160df77f4ac19dcd44b60b9c64c6 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 17:58:07 +0300 Subject: [PATCH 17/52] Change embeddings to be stored separately --- internal/core/services/messageService.go | 95 ++++++++++++++----- .../chatSessions/createChatSessionHandler.go | 14 --- internal/handlers/chatSessions/dto.go | 6 -- .../chatSessions/getChatSessionHandler.go | 40 ++++---- pkg/embeddings/embedder.go | 67 +++++++------ 5 files changed, 127 insertions(+), 95 deletions(-) diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index 9bee4a0..f7a1714 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -7,7 +7,9 @@ import ( "github.com/google/uuid" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/core/ports" + "github.com/loukaspe/jedi-team-challenge/internal/repositories" apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/helpers" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "github.com/openai/openai-go" "strings" @@ -19,7 +21,7 @@ type MessageServiceInterface interface { } type Embedder interface { - Embed(ctx context.Context, inputs []string) ([][]float64, error) + Embed(context.Context, []string) ([]*domain.Embeddings, error) } type VectorDB interface { @@ -72,21 +74,55 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI return nil, err } - embeddings, err := s.embedder.Embed(context.Background(), []string{ - "What do you know about Gen Z in Nashville", - }) + chatSession, err := s.chatSessionRepository.GetChatSession(ctx, initialMessage.ChatSessionID) if err != nil { return nil, err } - // we only have on text so we only care for the first embedding row - vectorToFloat32 := make([]float32, len(embeddings[0])) - for i, v := range embeddings[0] { - vectorToFloat32[i] = float32(v) + if chatSession.Title == "" { + title, err := s.generateTitleFromOpenAI(initialMessage.Content) + if err != nil { + return nil, err + } + + err = s.chatSessionRepository.UpdateChatSessionTitle(ctx, initialMessage.ChatSessionID, title) + if err != nil { + return nil, err + } + } + + domainEmbeddings, err := s.embedder.Embed(context.Background(), []string{initialMessage.Content}) + if err != nil { + return nil, err } + // we only have on text so we only care for the first embedding row + vectorToFloat32 := helpers.Float64ToFloat32(domainEmbeddings[0].Embeddings) + accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(context.Background(), vectorToFloat32) + answer, err := s.generateAnswerFromOpenAI(accumulatedTextFromSearch, initialMessage.Content) + if err != nil { + return nil, err + } + + replyMessage := &domain.Message{ + ChatSessionID: initialMessage.ChatSessionID, + Content: answer, + Sender: repositories.SYSTEM_SENDER, + } + + insertedMessageID, err := s.messageRepository.CreateMessage(ctx, replyMessage) + if err != nil { + return nil, err + } + + replyMessage.ID = insertedMessageID + + return replyMessage, nil +} + +func (s *MessageService) generateAnswerFromOpenAI(text []string, initialMessage string) (string, error) { prompt := fmt.Sprintf(`Use the following context to answer the question. Context: %s @@ -95,8 +131,8 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI %s Answer:`, - strings.Join(accumulatedTextFromSearch, "\n"), - initialMessage.Content, + strings.Join(text, "\n"), + initialMessage, ) chatCompletion, err := s.openAIClient.Chat.Completions.New(context.Background(), openai.ChatCompletionNewParams{ @@ -106,29 +142,38 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI Model: openai.ChatModelGPT4o, }) if err != nil { - return nil, err + return "", err } - answer := chatCompletion.Choices[0].Message.Content + if chatCompletion.Choices[0].Message.Content == "" { + return "", errors.New("received empty response from LLM") - if answer == "" { - // TODO: mh ta les apierrors - return nil, errors.New("received empty response from LLM") } - replyMessage := &domain.Message{ - ID: initialMessageID, - ChatSessionID: initialMessage.ChatSessionID, - Content: answer, - Sender: "SYSTEM", //TODO: constant - } + return chatCompletion.Choices[0].Message.Content, nil +} - insertedMessageID, err := s.messageRepository.CreateMessage(ctx, replyMessage) +func (s *MessageService) generateTitleFromOpenAI(initialMessage string) (string, error) { + prompt := fmt.Sprintf(`Summarize the following user message into a short, descriptive chat title (max 5 words): + %s + Answer:`, + initialMessage, + ) + + chatCompletion, err := s.openAIClient.Chat.Completions.New(context.Background(), openai.ChatCompletionNewParams{ + Messages: []openai.ChatCompletionMessageParamUnion{ + openai.UserMessage(prompt), + }, + Model: openai.ChatModelGPT4o, + }) if err != nil { - return nil, err + return "", err } - replyMessage.ID = insertedMessageID + if chatCompletion.Choices[0].Message.Content == "" { + return "", errors.New("received empty response from LLM") - return replyMessage, nil + } + + return chatCompletion.Choices[0].Message.Content, nil } diff --git a/internal/handlers/chatSessions/createChatSessionHandler.go b/internal/handlers/chatSessions/createChatSessionHandler.go index 4292fe0..b90b2b9 100644 --- a/internal/handlers/chatSessions/createChatSessionHandler.go +++ b/internal/handlers/chatSessions/createChatSessionHandler.go @@ -66,20 +66,6 @@ func (handler *CreateUserChatSessionHandler) CreateUserChatSessionController(w h return } - //err = json.NewDecoder(r.Body).Decode(chatSessionRequest) - //if err != nil { - // handler.logger.Error("Error in creating chat session", - // map[string]interface{}{ - // "errorMessage": err.Error(), - // }) - // - // response.ErrorMessage = "malformed create chat session request" - // - // handler.JsonResponse(w, http.StatusBadRequest, response) - // - // return - //} - insertedUUID, err := handler.ChatSessionService.CreateChatSession( ctx, &domain.ChatSession{ diff --git a/internal/handlers/chatSessions/dto.go b/internal/handlers/chatSessions/dto.go index ac3cfae..d01f335 100644 --- a/internal/handlers/chatSessions/dto.go +++ b/internal/handlers/chatSessions/dto.go @@ -66,13 +66,7 @@ func MessageResponseFromModel(msg *domain.Message) *MessageResponse { } } -//type CreateChatSessionRequest struct { -// UserID string `json:"chatSessionID"` -//} - type SendMessageRequest struct { - //UserID string `json:"chatSessionID"` - //SessionID string `json:"sessionId"` Content string `json:"content"` } diff --git a/internal/handlers/chatSessions/getChatSessionHandler.go b/internal/handlers/chatSessions/getChatSessionHandler.go index b4fd577..b1742a6 100644 --- a/internal/handlers/chatSessions/getChatSessionHandler.go +++ b/internal/handlers/chatSessions/getChatSessionHandler.go @@ -25,15 +25,15 @@ func NewGetChatSessionHandler( } } -// @Summary Gets all User's chat sessions -// @Description Gets all User's chat sessions -// @Security BearerAuth -// @Param user_id path int true "user id" -// @Success 201 {object} UserChatSessionsResponse -// @Failure 400 {object} UserChatSessionsResponse "Error in message payload" -// @Failure 401 {object} UserChatSessionsResponse "Authentication error" -// @Failure 500 {object} UserChatSessionsResponse "Internal Server Error" -// @Router /users/user_id/chat-sessions [get] +// @Summary Gets all User's chat sessions +// @Description Gets all User's chat sessions +// @Security BearerAuth +// @Param user_id path int true "user id" +// @Success 201 {object} UserChatSessionsResponse +// @Failure 400 {object} UserChatSessionsResponse "Error in message payload" +// @Failure 401 {object} UserChatSessionsResponse "Authentication error" +// @Failure 500 {object} UserChatSessionsResponse "Internal Server Error" +// @Router /users/user_id/chat-sessions [get] func (handler *GetChatSessionHandler) GetUserChatSessionsController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") @@ -79,12 +79,12 @@ func (handler *GetChatSessionHandler) GetUserChatSessionsController(w http.Respo } if err != nil { - handler.logger.Error("Error in getting user's favourite assets", + handler.logger.Error("Error in getting user's chat sessions", map[string]interface{}{ "errorMessage": err.Error(), }) - response.ErrorMessage = "error in getting user favourite assets" + response.ErrorMessage = "error in getting user chat sessions" handler.JsonUserChatSessionResponse(w, http.StatusInternalServerError, response) return @@ -94,15 +94,15 @@ func (handler *GetChatSessionHandler) GetUserChatSessionsController(w http.Respo handler.JsonUserChatSessionResponse(w, http.StatusOK, response) } -// @Summary Gets chat session -// @Description Gets chat session -// @Security BearerAuth -// @Param session_id path int true "session id" -// @Success 201 {object} ChatSessionResponse -// @Failure 400 {object} ChatSessionResponse "Error in message payload" -// @Failure 401 {object} ChatSessionResponse "Authentication error" -// @Failure 500 {object} ChatSessionResponse "Internal Server Error" -// @Router /chat-sessions/session_id [get] +// @Summary Gets chat session +// @Description Gets chat session +// @Security BearerAuth +// @Param session_id path int true "session id" +// @Success 201 {object} ChatSessionResponse +// @Failure 400 {object} ChatSessionResponse "Error in message payload" +// @Failure 401 {object} ChatSessionResponse "Authentication error" +// @Failure 500 {object} ChatSessionResponse "Internal Server Error" +// @Router /chat-sessions/session_id [get] func (handler *GetChatSessionHandler) GetChatSessionController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") diff --git a/pkg/embeddings/embedder.go b/pkg/embeddings/embedder.go index 1b55422..c2c0295 100644 --- a/pkg/embeddings/embedder.go +++ b/pkg/embeddings/embedder.go @@ -4,6 +4,7 @@ import ( "context" "errors" "fmt" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/openai/openai-go" "math" "time" @@ -23,43 +24,49 @@ func NewEmbeddingService(client *openai.Client, embeddingModel openai.EmbeddingM } } -func (s *EmbeddingService) Embed(ctx context.Context, inputs []string) ([][]float64, error) { +func (s *EmbeddingService) Embed(ctx context.Context, inputs []string) ([]*domain.Embeddings, error) { var resp *openai.CreateEmbeddingResponse var err error + domainEmbeddings := make([]*domain.Embeddings, 0) - for attempt := 0; attempt < maxEmbeddingRetries; attempt++ { - resp, err = s.OpenAIClient.Embeddings.New( - ctx, - openai.EmbeddingNewParams{ - Model: s.EmbeddingModel, - Input: openai.EmbeddingNewParamsInputUnion{OfArrayOfStrings: inputs}, - }, - ) + for _, input := range inputs { + for attempt := 0; attempt < maxEmbeddingRetries; attempt++ { + resp, err = s.OpenAIClient.Embeddings.New( + ctx, + openai.EmbeddingNewParams{ + Model: s.EmbeddingModel, + Input: openai.EmbeddingNewParamsInputUnion{OfString: openai.String(input)}, + }, + ) - if err == nil { - break + if err == nil { + break + } + + // see if it's an APIError with HTTP 429 + var apiErr *openai.Error + if errors.As(err, &apiErr) && apiErr.StatusCode == 429 { + wait := time.Duration(math.Pow(2, float64(attempt))) * time.Second + // TODO: log + fmt.Printf("Rate‐limited on attempt %d: waiting %s before retry…", attempt+1, wait) + time.Sleep(wait) + continue + } + // some other error – give up immediately + return nil, fmt.Errorf("failed embedding: %w", err) } - // see if it's an APIError with HTTP 429 - var apiErr *openai.Error - if errors.As(err, &apiErr) && apiErr.StatusCode == 429 { - wait := time.Duration(math.Pow(2, float64(attempt))) * time.Second - // TODO: log - fmt.Printf("Rate‐limited on attempt %d: waiting %s before retry…", attempt+1, wait) - time.Sleep(wait) - continue + if err != nil { + return nil, err } - // some other error – give up immediately - return nil, fmt.Errorf("failed embedding: %w", err) - } - if err != nil { - return nil, err - } - // Collect embeddings - embeddings := make([][]float64, len(resp.Data)) - for i, d := range resp.Data { - embeddings[i] = d.Embedding + for _, d := range resp.Data { + domainEmbeddings = append(domainEmbeddings, &domain.Embeddings{ + Embeddings: d.Embedding, + Text: input, + }) + } } - return embeddings, nil + + return domainEmbeddings, nil } From 354cff9600e2073e93136d891b84fd3ab3fc23bd Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 17:59:35 +0300 Subject: [PATCH 18/52] Small fix to embedding separately --- cmd/http/main.go | 19 +++---------------- 1 file changed, 3 insertions(+), 16 deletions(-) diff --git a/cmd/http/main.go b/cmd/http/main.go index a25e6d2..948ccce 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -6,7 +6,6 @@ import ( "github.com/google/uuid" "github.com/gorilla/mux" "github.com/joho/godotenv" - "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/repositories" "github.com/loukaspe/jedi-team-challenge/pkg/chunks" "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" @@ -79,12 +78,13 @@ func getDB() *gorm.DB { log.Fatal("cannot migrate user table") } - // TODO: remove + // hardcoded user just for testing purposes admin := repositories.User{ ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, Username: "loukas", Password: "loukastest", } + fmt.Printf("Seeding user: %s\n", admin.ID) err = db.Debug().Model(&repositories.User{}).Create(&admin).Error if err != nil { @@ -177,24 +177,11 @@ func inputKnowledgeBase(chunker *chunks.Chunker, embedder *embeddings.EmbeddingS fmt.Printf("Generated %d chunks\n", len(chunks)) ctx := context.Background() - embeddings, err := embedder.Embed(ctx, chunks) + domainEmbeddings, err := embedder.Embed(ctx, chunks) if err != nil { log.Fatalf("Embedding error: %v", err) } - // Output - for i, emb := range embeddings { - fmt.Printf("Chunk %d → %d-dim vector\n", i, len(emb)) - } - - domainEmbeddings := make([]*domain.Embeddings, 0, len(embeddings)) - for i, e := range embeddings { - domainEmbeddings = append(domainEmbeddings, &domain.Embeddings{ - Embeddings: e, - Text: chunks[i], - }) - } - count, err := pineconeVectorDB.StoreEmbeddings(ctx, domainEmbeddings) if err != nil { log.Fatalf("Failed to store embeddings: %v", err) From cd0411bba91e55dfd97a5e4eabb447593fc3cc67 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 20:02:06 +0300 Subject: [PATCH 19/52] Minor fix --- .../core/ports/chatSessionRepositoryInterface.go | 1 - internal/core/services/chatSessionService.go | 5 ----- .../core/services/chatSessionService.go | 14 -------------- 3 files changed, 20 deletions(-) diff --git a/internal/core/ports/chatSessionRepositoryInterface.go b/internal/core/ports/chatSessionRepositoryInterface.go index 6e92c3e..32b39dc 100644 --- a/internal/core/ports/chatSessionRepositoryInterface.go +++ b/internal/core/ports/chatSessionRepositoryInterface.go @@ -11,5 +11,4 @@ type ChatSessionRepositoryInterface interface { GetUserChatSessions(context.Context, uuid.UUID) ([]*domain.ChatSession, error) CreateChatSession(context.Context, *domain.ChatSession) (uuid.UUID, error) UpdateChatSessionTitle(context.Context, uuid.UUID, string) error - DeleteChatSession(context.Context, uuid.UUID) error } diff --git a/internal/core/services/chatSessionService.go b/internal/core/services/chatSessionService.go index 28b5953..935265c 100644 --- a/internal/core/services/chatSessionService.go +++ b/internal/core/services/chatSessionService.go @@ -13,7 +13,6 @@ type ChatSessionServiceInterface interface { GetUserChatSessions(context.Context, uuid.UUID) ([]*domain.ChatSession, error) CreateChatSession(context.Context, *domain.ChatSession) (uuid.UUID, error) UpdateChatSessionTitle(context.Context, uuid.UUID, string) error - DeleteChatSession(context.Context, uuid.UUID) error } type ChatSessionService struct { @@ -39,10 +38,6 @@ func (s ChatSessionService) UpdateChatSessionTitle(ctx context.Context, uuid uui return s.repository.UpdateChatSessionTitle(ctx, uuid, title) } -func (s ChatSessionService) DeleteChatSession(ctx context.Context, uuid uuid.UUID) error { - return s.repository.DeleteChatSession(ctx, uuid) -} - func (s ChatSessionService) GetChatSession(ctx context.Context, uuid uuid.UUID) (*domain.ChatSession, error) { return s.repository.GetChatSession(ctx, uuid) } diff --git a/mocks/mock_internal/core/services/chatSessionService.go b/mocks/mock_internal/core/services/chatSessionService.go index 3788ecd..a149647 100644 --- a/mocks/mock_internal/core/services/chatSessionService.go +++ b/mocks/mock_internal/core/services/chatSessionService.go @@ -56,20 +56,6 @@ func (mr *MockChatSessionServiceInterfaceMockRecorder) CreateChatSession(arg0, a return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "CreateChatSession", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).CreateChatSession), arg0, arg1) } -// DeleteChatSession mocks base method. -func (m *MockChatSessionServiceInterface) DeleteChatSession(arg0 context.Context, arg1 uuid.UUID) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "DeleteChatSession", arg0, arg1) - ret0, _ := ret[0].(error) - return ret0 -} - -// DeleteChatSession indicates an expected call of DeleteChatSession. -func (mr *MockChatSessionServiceInterfaceMockRecorder) DeleteChatSession(arg0, arg1 any) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "DeleteChatSession", reflect.TypeOf((*MockChatSessionServiceInterface)(nil).DeleteChatSession), arg0, arg1) -} - // GetChatSession mocks base method. func (m *MockChatSessionServiceInterface) GetChatSession(arg0 context.Context, arg1 uuid.UUID) (*domain.ChatSession, error) { m.ctrl.T.Helper() From 1974d8f4619a4fe5cc19a536ffba1614521f8fd2 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 20:02:32 +0300 Subject: [PATCH 20/52] Adds similarity search threshold --- pkg/vectordb/pinecone.go | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/pkg/vectordb/pinecone.go b/pkg/vectordb/pinecone.go index 6f78e22..663dd0b 100644 --- a/pkg/vectordb/pinecone.go +++ b/pkg/vectordb/pinecone.go @@ -13,13 +13,15 @@ type PineconeVectorDB struct { client *pinecone.Client index string topKResultsNumber int + threshold float32 } -func NewPineconeVectorDB(topKResultsNumber int, index string, client *pinecone.Client) *PineconeVectorDB { +func NewPineconeVectorDB(threshold float32, topKResultsNumber int, index string, client *pinecone.Client) *PineconeVectorDB { return &PineconeVectorDB{ topKResultsNumber: topKResultsNumber, index: index, client: client, + threshold: threshold, } } @@ -83,7 +85,11 @@ func (db *PineconeVectorDB) SemanticSearch(ctx context.Context, embeddings []flo var contextTexts []string for _, match := range res.Matches { text := match.Vector.Metadata.String() - contextTexts = append(contextTexts, text) + score := match.Score + + if score >= db.threshold { + contextTexts = append(contextTexts, text) + } } return contextTexts, nil From 6d6135b4e0302319f1cf3463fd53a54d5b4dded2 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 20:14:51 +0300 Subject: [PATCH 21/52] adds context and changes openai model --- cmd/http/main.go | 13 ++++++++++--- internal/core/services/messageService.go | 20 ++++++++++---------- 2 files changed, 20 insertions(+), 13 deletions(-) diff --git a/cmd/http/main.go b/cmd/http/main.go index 948ccce..d02d292 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -25,6 +25,7 @@ import ( ) func main() { + ctx := context.Background() getEnv() encoder := getEncoder() @@ -35,7 +36,7 @@ func main() { inputKnowledgeBase(chunker, embedder, pineconeVectorDB) - logger := logger.NewLogger(context.Background()) + logger := logger.NewLogger(ctx) router := mux.NewRouter() httpServer := &http.Server{ Addr: os.Getenv("SERVER_ADDR"), @@ -152,6 +153,12 @@ func getPineconeVectorDB() *vectordb.PineconeVectorDB { log.Fatal("Cannot read top k results number: ", err) } + similaritySearchThresholdAsString := os.Getenv("SIMILARITY_SEARCH_THRESHOLD") + similaritySearchThreshold, err := strconv.ParseFloat(similaritySearchThresholdAsString, 32) + if err != nil { + log.Fatal("Cannot read similarity search threshold: ", err) + } + pineconeClient, err := pinecone.NewClient(pinecone.NewClientParams{ ApiKey: os.Getenv("PINECONE_API_KEY"), }) @@ -159,6 +166,7 @@ func getPineconeVectorDB() *vectordb.PineconeVectorDB { log.Fatalf("Failed to create pinecone Client: %v", err) } return vectordb.NewPineconeVectorDB( + float32(similaritySearchThreshold), topKResultsNumber, os.Getenv("PINECONE_INDEX"), pineconeClient, @@ -166,7 +174,7 @@ func getPineconeVectorDB() *vectordb.PineconeVectorDB { } -func inputKnowledgeBase(chunker *chunks.Chunker, embedder *embeddings.EmbeddingService, pineconeVectorDB *vectordb.PineconeVectorDB) { +func inputKnowledgeBase(ctx context.Context, chunker *chunks.Chunker, embedder *embeddings.EmbeddingService, pineconeVectorDB *vectordb.PineconeVectorDB) { textBytes, err := os.ReadFile("./data.md") if err != nil { log.Fatal(err) @@ -176,7 +184,6 @@ func inputKnowledgeBase(chunker *chunks.Chunker, embedder *embeddings.EmbeddingS chunks := chunker.Chunk(text) fmt.Printf("Generated %d chunks\n", len(chunks)) - ctx := context.Background() domainEmbeddings, err := embedder.Embed(ctx, chunks) if err != nil { log.Fatalf("Embedding error: %v", err) diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index f7a1714..55ddaf6 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -80,7 +80,7 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI } if chatSession.Title == "" { - title, err := s.generateTitleFromOpenAI(initialMessage.Content) + title, err := s.generateTitleFromOpenAI(ctx, initialMessage.Content) if err != nil { return nil, err } @@ -91,7 +91,7 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI } } - domainEmbeddings, err := s.embedder.Embed(context.Background(), []string{initialMessage.Content}) + domainEmbeddings, err := s.embedder.Embed(ctx, []string{initialMessage.Content}) if err != nil { return nil, err } @@ -99,9 +99,9 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI // we only have on text so we only care for the first embedding row vectorToFloat32 := helpers.Float64ToFloat32(domainEmbeddings[0].Embeddings) - accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(context.Background(), vectorToFloat32) + accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(ctx, vectorToFloat32) - answer, err := s.generateAnswerFromOpenAI(accumulatedTextFromSearch, initialMessage.Content) + answer, err := s.generateAnswerFromOpenAI(ctx, accumulatedTextFromSearch, initialMessage.Content) if err != nil { return nil, err } @@ -122,7 +122,7 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI return replyMessage, nil } -func (s *MessageService) generateAnswerFromOpenAI(text []string, initialMessage string) (string, error) { +func (s *MessageService) generateAnswerFromOpenAI(ctx context.Context, text []string, initialMessage string) (string, error) { prompt := fmt.Sprintf(`Use the following context to answer the question. Context: %s @@ -135,11 +135,11 @@ func (s *MessageService) generateAnswerFromOpenAI(text []string, initialMessage initialMessage, ) - chatCompletion, err := s.openAIClient.Chat.Completions.New(context.Background(), openai.ChatCompletionNewParams{ + chatCompletion, err := s.openAIClient.Chat.Completions.New(ctx, openai.ChatCompletionNewParams{ Messages: []openai.ChatCompletionMessageParamUnion{ openai.UserMessage(prompt), }, - Model: openai.ChatModelGPT4o, + Model: openai.ChatModelGPT4_1Nano, }) if err != nil { return "", err @@ -153,18 +153,18 @@ func (s *MessageService) generateAnswerFromOpenAI(text []string, initialMessage return chatCompletion.Choices[0].Message.Content, nil } -func (s *MessageService) generateTitleFromOpenAI(initialMessage string) (string, error) { +func (s *MessageService) generateTitleFromOpenAI(ctx context.Context, initialMessage string) (string, error) { prompt := fmt.Sprintf(`Summarize the following user message into a short, descriptive chat title (max 5 words): %s Answer:`, initialMessage, ) - chatCompletion, err := s.openAIClient.Chat.Completions.New(context.Background(), openai.ChatCompletionNewParams{ + chatCompletion, err := s.openAIClient.Chat.Completions.New(ctx, openai.ChatCompletionNewParams{ Messages: []openai.ChatCompletionMessageParamUnion{ openai.UserMessage(prompt), }, - Model: openai.ChatModelGPT4o, + Model: openai.ChatModelGPT4_1Nano, }) if err != nil { return "", err From 436c12bb383c25af976d8211540080f999104d77 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 20:45:05 +0300 Subject: [PATCH 22/52] Adds history to the chat completion --- internal/core/services/messageService.go | 22 ++++++++++++++++++---- 1 file changed, 18 insertions(+), 4 deletions(-) diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index 55ddaf6..e8f7f14 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -101,7 +101,7 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(ctx, vectorToFloat32) - answer, err := s.generateAnswerFromOpenAI(ctx, accumulatedTextFromSearch, initialMessage.Content) + answer, err := s.generateAnswerFromOpenAI(ctx, accumulatedTextFromSearch, initialMessage.Content, chatSession.Messages) if err != nil { return nil, err } @@ -122,7 +122,7 @@ func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageI return replyMessage, nil } -func (s *MessageService) generateAnswerFromOpenAI(ctx context.Context, text []string, initialMessage string) (string, error) { +func (s *MessageService) generateAnswerFromOpenAI(ctx context.Context, text []string, initialMessage string, previousMessages []*domain.Message) (string, error) { prompt := fmt.Sprintf(`Use the following context to answer the question. Context: %s @@ -135,12 +135,26 @@ func (s *MessageService) generateAnswerFromOpenAI(ctx context.Context, text []st initialMessage, ) - chatCompletion, err := s.openAIClient.Chat.Completions.New(ctx, openai.ChatCompletionNewParams{ + completionParams := openai.ChatCompletionNewParams{ Messages: []openai.ChatCompletionMessageParamUnion{ openai.UserMessage(prompt), }, Model: openai.ChatModelGPT4_1Nano, - }) + } + + if len(previousMessages) >= 0 { + for _, msg := range previousMessages { + completionParams.Messages = append(completionParams.Messages, openai.ChatCompletionMessageParamUnion{ + OfAssistant: &openai.ChatCompletionAssistantMessageParam{ + Content: openai.ChatCompletionAssistantMessageParamContentUnion{ + OfString: openai.String(msg.Content), + }, + }, + }) + } + } + + chatCompletion, err := s.openAIClient.Chat.Completions.New(ctx, completionParams) if err != nil { return "", err } From fdddddf2d08db5ada5afef67fd6d1cc02d6ca1d3 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 20:45:18 +0300 Subject: [PATCH 23/52] Make the e2e for clear and specific --- script/e2e.sh | 24 +++++++++++++++++------- 1 file changed, 17 insertions(+), 7 deletions(-) diff --git a/script/e2e.sh b/script/e2e.sh index 3dea0c6..64a5cde 100644 --- a/script/e2e.sh +++ b/script/e2e.sh @@ -44,25 +44,35 @@ for i in {1..3}; do done # Step 3: Send the first message -message_content_1="what do you know about gen z in nashville" +message_content_1="what do you know about latino mobile gamers" response_message_1=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ --header "Content-Type: application/json" \ --header "Authorization: Bearer $token" \ --data "{\"content\":\"$message_content_1\"}") # Step 4: Send the second message -message_content_2="what do you know about latino mobile gamers" +message_content_2="do they use social media" response_message_2=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ --header "Content-Type: application/json" \ --header "Authorization: Bearer $token" \ --data "{\"content\":\"$message_content_2\"}") -# Step 5: Display the responses from sending the messages -echo "Message 1 sent to chat session $CHAT_SESSION_ID:" -echo "$response_message_1" +# Step 4.5: Send the third message +message_content_3="what social media do they use the most" +response_message_3=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ + --header "Content-Type: application/json" \ + --header "Authorization: Bearer $token" \ + --data "{\"content\":\"$message_content_3\"}") -echo "Message 2 sent to chat session $CHAT_SESSION_ID:" -echo "$response_message_2" +## Step 5: Display the responses from sending the messages +#echo "Message 1 sent to chat session $CHAT_SESSION_ID:" +#echo "$response_message_1" +# +#echo "Message 2 sent to chat session $CHAT_SESSION_ID:" +#echo "$response_message_2" +# +#echo "Message 3 sent to chat session $CHAT_SESSION_ID:" +#echo "$response_message_3" # Step 6: Fetch and display the whole chat session echo "Fetching the entire chat session $CHAT_SESSION_ID..." From 60c9f43e2729d0e168fdf6f9488e53f6b31860e5 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 21:43:51 +0300 Subject: [PATCH 24/52] Use only provided context, not chatgpts --- internal/core/services/messageService.go | 1 + 1 file changed, 1 insertion(+) diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index e8f7f14..e8d55ca 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -138,6 +138,7 @@ func (s *MessageService) generateAnswerFromOpenAI(ctx context.Context, text []st completionParams := openai.ChatCompletionNewParams{ Messages: []openai.ChatCompletionMessageParamUnion{ openai.UserMessage(prompt), + openai.SystemMessage("Use only the provided context for answering the question."), }, Model: openai.ChatModelGPT4_1Nano, } From a3d0f1cb14478f9f1e6f1a15abf8fbfc03100bd8 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 21:47:09 +0300 Subject: [PATCH 25/52] add context to knowledge base --- cmd/http/main.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cmd/http/main.go b/cmd/http/main.go index d02d292..104ac92 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -34,7 +34,7 @@ func main() { embedder := getEmbedder(&client) pineconeVectorDB := getPineconeVectorDB() - inputKnowledgeBase(chunker, embedder, pineconeVectorDB) + inputKnowledgeBase(ctx, chunker, embedder, pineconeVectorDB) logger := logger.NewLogger(ctx) router := mux.NewRouter() From e82a9bd2109b1663e0410bcf54a654b80ba33cb8 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 23:11:13 +0300 Subject: [PATCH 26/52] Adds submission of feedback for message --- internal/core/domain/chatSession.go | 1 + .../core/ports/messageRepositoryInterface.go | 1 + internal/core/services/messageService.go | 5 + internal/handlers/chatSessions/dto.go | 8 ++ .../chatSessions/submitFeedbackHandler.go | 133 ++++++++++++++++++ .../submitFeedbackHandler_test.go | 89 ++++++++++++ internal/repositories/chatSessionDao.go | 1 + .../messageRepositoryUpdateFeedback.go | 31 ++++ .../messageRepositoryUpdateFeedback_test.go | 77 ++++++++++ .../core/services/messageService.go | 90 ++++++++++++ pkg/server/routes.go | 2 + script/e2e.sh | 26 +++- 12 files changed, 458 insertions(+), 6 deletions(-) create mode 100644 internal/handlers/chatSessions/submitFeedbackHandler.go create mode 100644 internal/handlers/chatSessions/submitFeedbackHandler_test.go create mode 100644 internal/repositories/messageRepositoryUpdateFeedback.go create mode 100644 internal/repositories/messageRepositoryUpdateFeedback_test.go diff --git a/internal/core/domain/chatSession.go b/internal/core/domain/chatSession.go index 4ddc4a0..f581757 100644 --- a/internal/core/domain/chatSession.go +++ b/internal/core/domain/chatSession.go @@ -20,4 +20,5 @@ type Message struct { Sender string Content string CreatedAt time.Time + Feedback *string } diff --git a/internal/core/ports/messageRepositoryInterface.go b/internal/core/ports/messageRepositoryInterface.go index 10905e3..73d77d7 100644 --- a/internal/core/ports/messageRepositoryInterface.go +++ b/internal/core/ports/messageRepositoryInterface.go @@ -9,4 +9,5 @@ import ( type MessageRepositoryInterface interface { CreateMessage(context.Context, *domain.Message) (uuid.UUID, error) GetMessage(context.Context, uuid.UUID) (*domain.Message, error) + UpdateMessageFeedback(context.Context, uuid.UUID, string) error } diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index e8d55ca..69ed252 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -18,6 +18,7 @@ import ( type MessageServiceInterface interface { CreateMessage(context.Context, uuid.UUID, *domain.Message) (uuid.UUID, error) GetAnswerForMessage(context.Context, uuid.UUID) (*domain.Message, error) + UpdateMessageFeedback(context.Context, uuid.UUID, string) error } type Embedder interface { @@ -68,6 +69,10 @@ func (s MessageService) CreateMessage(ctx context.Context, userID uuid.UUID, mes return s.messageRepository.CreateMessage(ctx, message) } +func (s MessageService) UpdateMessageFeedback(ctx context.Context, messageID uuid.UUID, feedback string) error { + return s.messageRepository.UpdateMessageFeedback(ctx, messageID, feedback) +} + func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageID uuid.UUID) (*domain.Message, error) { initialMessage, err := s.messageRepository.GetMessage(ctx, initialMessageID) if err != nil { diff --git a/internal/handlers/chatSessions/dto.go b/internal/handlers/chatSessions/dto.go index d01f335..ca5d4d7 100644 --- a/internal/handlers/chatSessions/dto.go +++ b/internal/handlers/chatSessions/dto.go @@ -75,3 +75,11 @@ type SendMessageResponse struct { SystemMessage *MessageResponse `json:"systemMessage,omitempty"` ErrorMessage string `json:"errorMessage,omitempty"` } + +type SubmitFeedbackRequest struct { + Feedback string `json:"feedback"` +} + +type SubmitFeedbackResponse struct { + ErrorMessage string `json:"errorMessage,omitempty"` +} diff --git a/internal/handlers/chatSessions/submitFeedbackHandler.go b/internal/handlers/chatSessions/submitFeedbackHandler.go new file mode 100644 index 0000000..1c85b8e --- /dev/null +++ b/internal/handlers/chatSessions/submitFeedbackHandler.go @@ -0,0 +1,133 @@ +package chatSessions + +import ( + "encoding/json" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" +) + +type SubmitFeedbackHandler struct { + MessageService services.MessageServiceInterface + logger logger.LoggerInterface +} + +func NewSubmitFeedbackHandler( + service services.MessageServiceInterface, + logger logger.LoggerInterface, +) *SubmitFeedbackHandler { + return &SubmitFeedbackHandler{ + MessageService: service, + logger: logger, + } +} + +// @Summary Submits a feedback to a message +// @Description Submits a feedback to a message +// @Security BearerAuth +// @Param SubmitFeedbackRequest body SubmitFeedbackRequest true "request body" +// @Param message_id body int true "message_id" +// @Success 201 {object} SubmitFeedbackResponse +// @Failure 400 {object} SubmitFeedbackResponse "Error in message payload" +// @Failure 401 {object} SubmitFeedbackResponse "Authentication error" +// @Failure 500 {object} SubmitFeedbackResponse "Internal Server Error" +// @Router /users/user_id/chat-sessions/session_id/messages [post] +func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + + ctx := r.Context() + + var err error + response := &SubmitFeedbackResponse{} + request := &SubmitFeedbackRequest{} + + messageIDAsString := mux.Vars(r)["message_id"] + if messageIDAsString == "" { + response.ErrorMessage = "missing message id" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + messageID, err := uuid.Parse(messageIDAsString) + if err != nil { + handler.logger.Error("Error in submitting feedback", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed message uuid" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + err = json.NewDecoder(r.Body).Decode(request) + if err != nil { + handler.logger.Error("Error in submitting feedback", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed submitting feedback request" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + err = handler.MessageService.UpdateMessageFeedback( + ctx, + messageID, + request.Feedback, + ) + + if resourceNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": resourceNotFound.Unwrap(), + }) + + response.ErrorMessage = err.Error() + handler.JsonResponse(w, http.StatusNotFound, response) + + return + } + + if err != nil { + handler.logger.Error("Error in sending message", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "error in sending message" + handler.JsonResponse(w, http.StatusInternalServerError, response) + + return + } + + w.WriteHeader(http.StatusCreated) +} + +func (handler *SubmitFeedbackHandler) JsonResponse( + w http.ResponseWriter, + statusCode int, + response *SubmitFeedbackResponse, +) { + w.WriteHeader(statusCode) + err := json.NewEncoder(w).Encode(response) + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + response.ErrorMessage = "error in sending message - json response" + + handler.logger.Error("Error in sending message - json response", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + } +} diff --git a/internal/handlers/chatSessions/submitFeedbackHandler_test.go b/internal/handlers/chatSessions/submitFeedbackHandler_test.go new file mode 100644 index 0000000..04adf36 --- /dev/null +++ b/internal/handlers/chatSessions/submitFeedbackHandler_test.go @@ -0,0 +1,89 @@ +package chatSessions + +import ( + "bytes" + "context" + "encoding/json" + "github.com/google/uuid" + "github.com/gorilla/mux" + mock_services "github.com/loukaspe/jedi-team-challenge/mocks/mock_internal/core/services" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/stretchr/testify/assert" + "go.uber.org/mock/gomock" + "net/http/httptest" + "testing" +) + +func TestUpdateMessageFeedbackHandler_UpdateMessageFeedbackController(t *testing.T) { + logger := logger.NewLogger(context.Background()) + mockCtrl := gomock.NewController(t) + defer mockCtrl.Finish() + + mockService := mock_services.NewMockMessageServiceInterface(mockCtrl) + + type args struct { + userID uuid.UUID + chatSessionID uuid.UUID + messageID uuid.UUID + feedback string + } + + tests := []struct { + name string + args args + expectedStatusCode int + }{ + { + name: "valid", + args: args{ + messageID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + chatSessionID: uuid.UUID{0x32, 0x34, 0x56, 0x78}, + userID: uuid.UUID{0x42, 0x34, 0x56, 0x78}, + feedback: "abla", + }, + expectedStatusCode: 201, + }, + } + + //TODO: user mismatch sto feedback + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + mockRequest := httptest.NewRequest( + "POST", + "/users/"+tt.args.messageID.String()+"/chat-sessions/"+tt.args.chatSessionID.String()+"/messages/"+tt.args.messageID.String(), + bytes.NewBuffer( + json.RawMessage(`{"feedback":"abla"}`), + ), + ) + + vars := map[string]string{ + "user_id": tt.args.userID.String(), + "chat_session_id": tt.args.chatSessionID.String(), + "message_id": tt.args.messageID.String(), + } + mockRequest = mux.SetURLVars(mockRequest, vars) + + mockRequest.Header.Set("Content-Type", "application/json") + mockResponseRecorder := httptest.NewRecorder() + + mockService.EXPECT().UpdateMessageFeedback( + gomock.Any(), + tt.args.messageID, + tt.args.feedback, + ).Return(nil) + + handler := &SubmitFeedbackHandler{ + MessageService: mockService, + logger: logger, + } + sut := handler.SubmitFeedbackController + + sut(mockResponseRecorder, mockRequest) + + mockResponse := mockResponseRecorder.Result() + actualStatusCode := mockResponse.StatusCode + + assert.Equal(t, tt.expectedStatusCode, actualStatusCode) + }) + } +} diff --git a/internal/repositories/chatSessionDao.go b/internal/repositories/chatSessionDao.go index 462c1d6..05464f4 100644 --- a/internal/repositories/chatSessionDao.go +++ b/internal/repositories/chatSessionDao.go @@ -23,4 +23,5 @@ type Message struct { Sender string `gorm:"type:text;not null"` Content string `gorm:"type:text;not null"` CreatedAt time.Time `gorm:"not null"` + Feedback *string `gorm:"type:text;null"` } diff --git a/internal/repositories/messageRepositoryUpdateFeedback.go b/internal/repositories/messageRepositoryUpdateFeedback.go new file mode 100644 index 0000000..5467ba5 --- /dev/null +++ b/internal/repositories/messageRepositoryUpdateFeedback.go @@ -0,0 +1,31 @@ +package repositories + +import ( + "context" + "errors" + "github.com/google/uuid" + apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "gorm.io/gorm" +) + +func (repo *MessageRepository) UpdateMessageFeedback( + ctx context.Context, + uuid uuid.UUID, + feedback string, +) error { + var err error + + err = repo.db.WithContext(ctx). + Model(Message{}). + Where("id = ?", uuid). + Update("feedback", feedback).Error + + if err == gorm.ErrRecordNotFound { + // TODO: apierrors + return apierrors.ResourceNotFoundErrorWrapper{ + OriginalError: errors.New("messageID " + uuid.String() + " not found"), + } + } + + return err +} diff --git a/internal/repositories/messageRepositoryUpdateFeedback_test.go b/internal/repositories/messageRepositoryUpdateFeedback_test.go new file mode 100644 index 0000000..baddd8b --- /dev/null +++ b/internal/repositories/messageRepositoryUpdateFeedback_test.go @@ -0,0 +1,77 @@ +package repositories + +import ( + "context" + "github.com/DATA-DOG/go-sqlmock" + "github.com/google/uuid" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/stretchr/testify/assert" + "gorm.io/driver/postgres" + "gorm.io/gorm" + "regexp" + "testing" +) + +func TestChatRepository_UpdateMessageFeedback(t *testing.T) { + feedback := "ablabla" + + db, mockDb, err := sqlmock.New() + if err != nil { + t.Error(err.Error()) + } + defer db.Close() + + gormDb, err := gorm.Open(postgres.New(postgres.Config{Conn: db})) + + type args struct { + uuid uuid.UUID + feedback string + } + tests := []struct { + name string + args args + mockSqlMessageQueryExpected string + expected *domain.Message + }{ + { + name: "valid", + args: args{ + uuid: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + feedback: feedback, + }, + mockSqlMessageQueryExpected: `UPDATE "messages" SET "feedback"=$1 WHERE id = $2`, + expected: &domain.Message{ + ID: uuid.UUID{0x12, 0x34, 0x56, 0x78}, + ChatSessionID: uuid.UUID{0x22, 0x34, 0x56, 0x78}, + Sender: "user1", + Content: "Hello, world!", + Feedback: &feedback, + }, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + repo := &MessageRepository{ + db: gormDb, + } + + mockDb.ExpectBegin() + mockDb.ExpectExec(regexp.QuoteMeta(tt.mockSqlMessageQueryExpected)). + WithArgs(tt.args.feedback, tt.args.uuid). + WillReturnResult(sqlmock.NewResult(0, 1)) + mockDb.ExpectCommit() + + err := repo.UpdateMessageFeedback(context.Background(), tt.args.uuid, tt.args.feedback) + if err != nil { + t.Errorf("UpdateMessageFeedback() error = %v", err) + return + } + + assert.Nil(t, err) + + if err = mockDb.ExpectationsWereMet(); err != nil { + t.Errorf("there were unfulfilled expections: %s", err) + } + }) + } +} diff --git a/mocks/mock_internal/core/services/messageService.go b/mocks/mock_internal/core/services/messageService.go index aaa8dff..036e584 100644 --- a/mocks/mock_internal/core/services/messageService.go +++ b/mocks/mock_internal/core/services/messageService.go @@ -70,3 +70,93 @@ func (mr *MockMessageServiceInterfaceMockRecorder) GetAnswerForMessage(arg0, arg mr.mock.ctrl.T.Helper() return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetAnswerForMessage", reflect.TypeOf((*MockMessageServiceInterface)(nil).GetAnswerForMessage), arg0, arg1) } + +// UpdateMessageFeedback mocks base method. +func (m *MockMessageServiceInterface) UpdateMessageFeedback(arg0 context.Context, arg1 uuid.UUID, arg2 string) error { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "UpdateMessageFeedback", arg0, arg1, arg2) + ret0, _ := ret[0].(error) + return ret0 +} + +// UpdateMessageFeedback indicates an expected call of UpdateMessageFeedback. +func (mr *MockMessageServiceInterfaceMockRecorder) UpdateMessageFeedback(arg0, arg1, arg2 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "UpdateMessageFeedback", reflect.TypeOf((*MockMessageServiceInterface)(nil).UpdateMessageFeedback), arg0, arg1, arg2) +} + +// MockEmbedder is a mock of Embedder interface. +type MockEmbedder struct { + ctrl *gomock.Controller + recorder *MockEmbedderMockRecorder +} + +// MockEmbedderMockRecorder is the mock recorder for MockEmbedder. +type MockEmbedderMockRecorder struct { + mock *MockEmbedder +} + +// NewMockEmbedder creates a new mock instance. +func NewMockEmbedder(ctrl *gomock.Controller) *MockEmbedder { + mock := &MockEmbedder{ctrl: ctrl} + mock.recorder = &MockEmbedderMockRecorder{mock} + return mock +} + +// EXPECT returns an object that allows the caller to indicate expected use. +func (m *MockEmbedder) EXPECT() *MockEmbedderMockRecorder { + return m.recorder +} + +// Embed mocks base method. +func (m *MockEmbedder) Embed(arg0 context.Context, arg1 []string) ([]*domain.Embeddings, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "Embed", arg0, arg1) + ret0, _ := ret[0].([]*domain.Embeddings) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// Embed indicates an expected call of Embed. +func (mr *MockEmbedderMockRecorder) Embed(arg0, arg1 any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Embed", reflect.TypeOf((*MockEmbedder)(nil).Embed), arg0, arg1) +} + +// MockVectorDB is a mock of VectorDB interface. +type MockVectorDB struct { + ctrl *gomock.Controller + recorder *MockVectorDBMockRecorder +} + +// MockVectorDBMockRecorder is the mock recorder for MockVectorDB. +type MockVectorDBMockRecorder struct { + mock *MockVectorDB +} + +// NewMockVectorDB creates a new mock instance. +func NewMockVectorDB(ctrl *gomock.Controller) *MockVectorDB { + mock := &MockVectorDB{ctrl: ctrl} + mock.recorder = &MockVectorDBMockRecorder{mock} + return mock +} + +// EXPECT returns an object that allows the caller to indicate expected use. +func (m *MockVectorDB) EXPECT() *MockVectorDBMockRecorder { + return m.recorder +} + +// SemanticSearch mocks base method. +func (m *MockVectorDB) SemanticSearch(ctx context.Context, embeddings []float32) ([]string, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "SemanticSearch", ctx, embeddings) + ret0, _ := ret[0].([]string) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// SemanticSearch indicates an expected call of SemanticSearch. +func (mr *MockVectorDBMockRecorder) SemanticSearch(ctx, embeddings any) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SemanticSearch", reflect.TypeOf((*MockVectorDB)(nil).SemanticSearch), ctx, embeddings) +} diff --git a/pkg/server/routes.go b/pkg/server/routes.go index a6baff0..0d22ebe 100644 --- a/pkg/server/routes.go +++ b/pkg/server/routes.go @@ -55,10 +55,12 @@ func (s *Server) initializeRoutes() { createChatSessionHandler := chatSessions.NewCreateUserChatSessionHandler(chatSessionService, s.logger) getChatSessionHandler := chatSessions.NewGetChatSessionHandler(chatSessionService, s.logger) sendMessageHandler := chatSessions.NewSendMessageHandler(messageService, s.logger) + submitFeedbackHandler := chatSessions.NewSubmitFeedbackHandler(messageService, s.logger) protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionController).Methods("POST") protected.HandleFunc("/users/{user_id}/chat-sessions", getChatSessionHandler.GetUserChatSessionsController).Methods("GET") protected.HandleFunc("/users/{user_id}/chat-sessions/{session_id}/messages", sendMessageHandler.SendMessageController).Methods("POST") + protected.HandleFunc("/users/{user_id}/chat-sessions/{session_id}/messages/{message_id}/feedback", submitFeedbackHandler.SubmitFeedbackController).Methods("POST") protected.HandleFunc("/chat-sessions/{session_id}", getChatSessionHandler.GetChatSessionController).Methods("GET") diff --git a/script/e2e.sh b/script/e2e.sh index 64a5cde..71b05ae 100644 --- a/script/e2e.sh +++ b/script/e2e.sh @@ -35,12 +35,12 @@ for i in {1..3}; do fi # Check if the session creation was successful -# if [[ $(echo "$response" | jq -r '.status') == "success" ]]; then -# echo "Chat session $i created successfully." -# else -# echo "Error creating chat session $i." -# exit -# fi + # if [[ $(echo "$response" | jq -r '.status') == "success" ]]; then + # echo "Chat session $i created successfully." + # else + # echo "Error creating chat session $i." + # exit + # fi done # Step 3: Send the first message @@ -64,6 +64,8 @@ response_message_3=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/ --header "Authorization: Bearer $token" \ --data "{\"content\":\"$message_content_3\"}") +system_message_id_3=$(echo "$response_message_3" | jq -r '.systemMessage.id') + ## Step 5: Display the responses from sending the messages #echo "Message 1 sent to chat session $CHAT_SESSION_ID:" #echo "$response_message_1" @@ -82,3 +84,15 @@ chat_session_response=$(curl -s --location "$BASE_URL/chat-sessions/$CHAT_SESSIO # Display the full chat session echo "Full Chat Session $CHAT_SESSION_ID:" echo "$chat_session_response" + +feedback_message="The force is not strong with that message." +feedback_response=$(curl -s --location --write-out "%{http_code}" --request POST "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages/$system_message_id_3/feedback" \ + --header "Authorization: Bearer $token" \ + --data "{\"feedback\":\"$feedback_message\"}") + +if [ "$feedback_response" -eq 201 ]; then + echo "Feedback OK" +else + echo "Error: Feedback submission failed. Status code: $feedback_response" + exit 1 +fi From a4f6a87fa7e8da36fd3c91d89cfcc00c4495573f Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 23:18:05 +0300 Subject: [PATCH 27/52] Small fixes --- internal/core/services/messageService.go | 26 ++++++++++++++----- .../chatSessions/createChatSessionHandler.go | 4 +-- .../createChatSessionHandler_test.go | 4 +-- .../chatSessions/getChatSessionHandler.go | 6 ++--- .../chatSessions/sendMessageHandler.go | 8 +++--- .../chatSessions/submitFeedbackHandler.go | 18 ++++++++++--- .../repositories/chatSessionRepositoryGet.go | 6 ++--- .../chatSessionRepositoryGet_test.go | 4 +-- .../chatSessionRepositoryUpdate.go | 4 +-- .../chatSessionRepositoryUpdate_test.go | 4 +-- internal/repositories/messageRepositoryGet.go | 4 +-- .../messageRepositoryUpdateFeedback.go | 6 ++--- pkg/embeddings/embedder.go | 5 +--- 13 files changed, 61 insertions(+), 38 deletions(-) diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index 69ed252..f2eef95 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -8,7 +8,7 @@ import ( "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/core/ports" "github.com/loukaspe/jedi-team-challenge/internal/repositories" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/helpers" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "github.com/openai/openai-go" @@ -56,24 +56,38 @@ func NewMessageService( } } -func (s MessageService) CreateMessage(ctx context.Context, userID uuid.UUID, message *domain.Message) (uuid.UUID, error) { +func (s *MessageService) CreateMessage(ctx context.Context, userID uuid.UUID, message *domain.Message) (uuid.UUID, error) { chatSession, err := s.chatSessionRepository.GetChatSession(ctx, message.ChatSessionID) if err != nil { return uuid.Nil, err } if chatSession.UserID != userID { - return uuid.Nil, apierrors.NewUserMismatchError(message.ChatSessionID.String(), userID.String()) + return uuid.Nil, customerrors.NewUserMismatchError(message.ChatSessionID.String(), userID.String()) } return s.messageRepository.CreateMessage(ctx, message) } -func (s MessageService) UpdateMessageFeedback(ctx context.Context, messageID uuid.UUID, feedback string) error { - return s.messageRepository.UpdateMessageFeedback(ctx, messageID, feedback) +func (s *MessageService) UpdateMessageFeedback(ctx context.Context, message *domain.Message, userID uuid.UUID) error { + chatSession, err := s.chatSessionRepository.GetChatSession(ctx, message.ChatSessionID) + if err != nil { + return err + } + + _, err = s.chatSessionRepository.GetChatSession(ctx, message.ChatSessionID) + if err != nil { + return err + } + + if chatSession.UserID != userID { + return customerrors.NewUserMismatchError(message.ChatSessionID.String(), userID.String()) + } + + return s.messageRepository.UpdateMessageFeedback(ctx, message.ID, *message.Feedback) } -func (s MessageService) GetAnswerForMessage(ctx context.Context, initialMessageID uuid.UUID) (*domain.Message, error) { +func (s *MessageService) GetAnswerForMessage(ctx context.Context, initialMessageID uuid.UUID) (*domain.Message, error) { initialMessage, err := s.messageRepository.GetMessage(ctx, initialMessageID) if err != nil { return nil, err diff --git a/internal/handlers/chatSessions/createChatSessionHandler.go b/internal/handlers/chatSessions/createChatSessionHandler.go index b90b2b9..81f99ba 100644 --- a/internal/handlers/chatSessions/createChatSessionHandler.go +++ b/internal/handlers/chatSessions/createChatSessionHandler.go @@ -6,7 +6,7 @@ import ( "github.com/gorilla/mux" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/core/services" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "net/http" ) @@ -72,7 +72,7 @@ func (handler *CreateUserChatSessionHandler) CreateUserChatSessionController(w h UserID: userId, }, ) - if userNotFoundError, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + if userNotFoundError, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { handler.logger.Error("Error in creating chat session", map[string]interface{}{ "errorMessage": userNotFoundError.Unwrap(), diff --git a/internal/handlers/chatSessions/createChatSessionHandler_test.go b/internal/handlers/chatSessions/createChatSessionHandler_test.go index 90099d7..0dd444d 100644 --- a/internal/handlers/chatSessions/createChatSessionHandler_test.go +++ b/internal/handlers/chatSessions/createChatSessionHandler_test.go @@ -8,7 +8,7 @@ import ( "github.com/gorilla/mux" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" mock_services "github.com/loukaspe/jedi-team-challenge/mocks/mock_internal/core/services" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "github.com/stretchr/testify/assert" "go.uber.org/mock/gomock" @@ -205,7 +205,7 @@ func TestCreateUserChatSessionHandler_CreateUserChatSessionAssetControllerHasSer userId: uuid.UUID{0x12, 0x34, 0x56, 0x78}, }, mockServiceInsertedID: uuid.UUID{}, - mockServiceResponseError: apierrors.ResourceNotFoundErrorWrapper{ + mockServiceResponseError: customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("user id 667 not found"), }, expected: json.RawMessage(`{} diff --git a/internal/handlers/chatSessions/getChatSessionHandler.go b/internal/handlers/chatSessions/getChatSessionHandler.go index b1742a6..89fbaf6 100644 --- a/internal/handlers/chatSessions/getChatSessionHandler.go +++ b/internal/handlers/chatSessions/getChatSessionHandler.go @@ -5,7 +5,7 @@ import ( "github.com/google/uuid" "github.com/gorilla/mux" "github.com/loukaspe/jedi-team-challenge/internal/core/services" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "net/http" ) @@ -66,7 +66,7 @@ func (handler *GetChatSessionHandler) GetUserChatSessionsController(w http.Respo } usersChatSessions, err := handler.ChatSessionService.GetUserChatSessions(ctx, userId) - if userNotFoundError, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + if userNotFoundError, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { handler.logger.Error("Error in getting users chat sessions", map[string]interface{}{ "errorMessage": userNotFoundError.Unwrap(), @@ -135,7 +135,7 @@ func (handler *GetChatSessionHandler) GetChatSessionController(w http.ResponseWr } chatSession, err := handler.ChatSessionService.GetChatSession(ctx, sessionID) - if chatSessionNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + if chatSessionNotFound, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { handler.logger.Error("Error in getting chat session", map[string]interface{}{ "errorMessage": chatSessionNotFound.Unwrap(), diff --git a/internal/handlers/chatSessions/sendMessageHandler.go b/internal/handlers/chatSessions/sendMessageHandler.go index a36e916..903d13d 100644 --- a/internal/handlers/chatSessions/sendMessageHandler.go +++ b/internal/handlers/chatSessions/sendMessageHandler.go @@ -7,7 +7,7 @@ import ( "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/core/services" "github.com/loukaspe/jedi-team-challenge/internal/repositories" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "net/http" ) @@ -119,7 +119,7 @@ func (handler *SendMessageHandler) SendMessageController(w http.ResponseWriter, domainMessage, ) - if resourceNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + if resourceNotFound, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { handler.logger.Error("Error in sending message", map[string]interface{}{ "errorMessage": resourceNotFound.Unwrap(), @@ -131,7 +131,7 @@ func (handler *SendMessageHandler) SendMessageController(w http.ResponseWriter, return } - if userMismatchError, ok := err.(apierrors.UserMismatchError); ok { + if userMismatchError, ok := err.(customerrors.UserMismatchError); ok { handler.logger.Error("Error in sending message", map[string]interface{}{ "errorMessage": userMismatchError.Error(), @@ -158,7 +158,7 @@ func (handler *SendMessageHandler) SendMessageController(w http.ResponseWriter, domainMessage.ID = insertedUUID replyMessage, err := handler.MessageService.GetAnswerForMessage(ctx, insertedUUID) - if resourceNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { + if resourceNotFound, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { handler.logger.Error("Error in replying to message", map[string]interface{}{ "errorMessage": resourceNotFound.Unwrap(), diff --git a/internal/handlers/chatSessions/submitFeedbackHandler.go b/internal/handlers/chatSessions/submitFeedbackHandler.go index 1c85b8e..9345ffa 100644 --- a/internal/handlers/chatSessions/submitFeedbackHandler.go +++ b/internal/handlers/chatSessions/submitFeedbackHandler.go @@ -5,7 +5,7 @@ import ( "github.com/google/uuid" "github.com/gorilla/mux" "github.com/loukaspe/jedi-team-challenge/internal/core/services" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "net/http" ) @@ -87,8 +87,8 @@ func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWr request.Feedback, ) - if resourceNotFound, ok := err.(apierrors.ResourceNotFoundErrorWrapper); ok { - handler.logger.Error("Error in sending message", + if resourceNotFound, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Error in submitting feedback", map[string]interface{}{ "errorMessage": resourceNotFound.Unwrap(), }) @@ -99,6 +99,18 @@ func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWr return } + if userMismatchError, ok := err.(customerrors.UserMismatchError); ok { + handler.logger.Error("Error in submitting feedback", + map[string]interface{}{ + "errorMessage": userMismatchError.Error(), + }) + + response.ErrorMessage = err.Error() + handler.JsonResponse(w, http.StatusForbidden, response) + + return + } + if err != nil { handler.logger.Error("Error in sending message", map[string]interface{}{ diff --git a/internal/repositories/chatSessionRepositoryGet.go b/internal/repositories/chatSessionRepositoryGet.go index b3d5447..75e9334 100644 --- a/internal/repositories/chatSessionRepositoryGet.go +++ b/internal/repositories/chatSessionRepositoryGet.go @@ -5,7 +5,7 @@ import ( "errors" "github.com/google/uuid" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "gorm.io/gorm" ) @@ -23,7 +23,7 @@ func (repo *ChatSessionRepository) GetChatSession( Take(&modelChatSession).Error if err == gorm.ErrRecordNotFound { - return &domain.ChatSession{}, apierrors.ResourceNotFoundErrorWrapper{ + return &domain.ChatSession{}, customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("chatSessionID " + uuid.String() + " not found"), } } @@ -67,7 +67,7 @@ func (repo *ChatSessionRepository) GetUserChatSessions( Find(&modelChatSessions).Error if err == gorm.ErrRecordNotFound { - return []*domain.ChatSession{}, apierrors.ResourceNotFoundErrorWrapper{ + return []*domain.ChatSession{}, customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("user uuid " + uuid.String() + " not found"), } } diff --git a/internal/repositories/chatSessionRepositoryGet_test.go b/internal/repositories/chatSessionRepositoryGet_test.go index a7a9664..e07bde4 100644 --- a/internal/repositories/chatSessionRepositoryGet_test.go +++ b/internal/repositories/chatSessionRepositoryGet_test.go @@ -6,7 +6,7 @@ import ( "github.com/DATA-DOG/go-sqlmock" "github.com/google/uuid" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/stretchr/testify/assert" "gorm.io/driver/postgres" "gorm.io/gorm" @@ -171,7 +171,7 @@ func TestChatRepository_GetChatSessionWithError(t *testing.T) { }, mockSqlChatQueryExpected: `SELECT * FROM "chat_sessions" WHERE id = $1 LIMIT $2`, mockSqlErrorReturned: gorm.ErrRecordNotFound, - expectedError: apierrors.ResourceNotFoundErrorWrapper{ + expectedError: customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("chatSessionID 12345678-0000-0000-0000-000000000000 not found"), }, }, diff --git a/internal/repositories/chatSessionRepositoryUpdate.go b/internal/repositories/chatSessionRepositoryUpdate.go index b43642f..4bbc6e6 100644 --- a/internal/repositories/chatSessionRepositoryUpdate.go +++ b/internal/repositories/chatSessionRepositoryUpdate.go @@ -4,7 +4,7 @@ import ( "context" "errors" "github.com/google/uuid" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "gorm.io/gorm" ) @@ -18,7 +18,7 @@ func (repo *ChatSessionRepository) UpdateChatSessionTitle( Update("title", title).Error if err == gorm.ErrRecordNotFound { - return apierrors.ResourceNotFoundErrorWrapper{ + return customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("chatSessionID " + uuid.String() + " not found"), } } diff --git a/internal/repositories/chatSessionRepositoryUpdate_test.go b/internal/repositories/chatSessionRepositoryUpdate_test.go index 0bc79da..1d8202f 100644 --- a/internal/repositories/chatSessionRepositoryUpdate_test.go +++ b/internal/repositories/chatSessionRepositoryUpdate_test.go @@ -5,7 +5,7 @@ import ( "errors" "github.com/DATA-DOG/go-sqlmock" "github.com/google/uuid" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/stretchr/testify/assert" "gorm.io/driver/postgres" "gorm.io/gorm" @@ -108,7 +108,7 @@ func TestChatRepository_UpdateChatSessionTitleWithError(t *testing.T) { }, mockSqlChatQueryExpected: `UPDATE "chat_sessions" SET "title"=$1,"updated_at"=$2 WHERE id = $3`, mockSqlErrorReturned: gorm.ErrRecordNotFound, - expectedError: apierrors.ResourceNotFoundErrorWrapper{ + expectedError: customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("chatSessionID 12345678-0000-0000-0000-000000000000 not found"), }, }, diff --git a/internal/repositories/messageRepositoryGet.go b/internal/repositories/messageRepositoryGet.go index c2b27d8..90a748d 100644 --- a/internal/repositories/messageRepositoryGet.go +++ b/internal/repositories/messageRepositoryGet.go @@ -5,7 +5,7 @@ import ( "errors" "github.com/google/uuid" "github.com/loukaspe/jedi-team-challenge/internal/core/domain" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "gorm.io/gorm" ) @@ -22,7 +22,7 @@ func (repo *MessageRepository) GetMessage( Take(&modelMessage).Error if err == gorm.ErrRecordNotFound { - return &domain.Message{}, apierrors.ResourceNotFoundErrorWrapper{ + return &domain.Message{}, customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("messageID " + uuid.String() + " not found"), } } diff --git a/internal/repositories/messageRepositoryUpdateFeedback.go b/internal/repositories/messageRepositoryUpdateFeedback.go index 5467ba5..9f21729 100644 --- a/internal/repositories/messageRepositoryUpdateFeedback.go +++ b/internal/repositories/messageRepositoryUpdateFeedback.go @@ -4,7 +4,7 @@ import ( "context" "errors" "github.com/google/uuid" - apierrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "gorm.io/gorm" ) @@ -21,8 +21,8 @@ func (repo *MessageRepository) UpdateMessageFeedback( Update("feedback", feedback).Error if err == gorm.ErrRecordNotFound { - // TODO: apierrors - return apierrors.ResourceNotFoundErrorWrapper{ + // TODO: customerrors + return customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("messageID " + uuid.String() + " not found"), } } diff --git a/pkg/embeddings/embedder.go b/pkg/embeddings/embedder.go index c2c0295..54865fd 100644 --- a/pkg/embeddings/embedder.go +++ b/pkg/embeddings/embedder.go @@ -43,16 +43,13 @@ func (s *EmbeddingService) Embed(ctx context.Context, inputs []string) ([]*domai break } - // see if it's an APIError with HTTP 429 var apiErr *openai.Error if errors.As(err, &apiErr) && apiErr.StatusCode == 429 { wait := time.Duration(math.Pow(2, float64(attempt))) * time.Second - // TODO: log - fmt.Printf("Rate‐limited on attempt %d: waiting %s before retry…", attempt+1, wait) time.Sleep(wait) continue } - // some other error – give up immediately + return nil, fmt.Errorf("failed embedding: %w", err) } From 62bce25f87ad34131398060d9c05f371d7f7658f Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 23:26:43 +0300 Subject: [PATCH 28/52] Fix e2e test code --- script/e2e.sh | 37 +++++++++++++------------------------ 1 file changed, 13 insertions(+), 24 deletions(-) diff --git a/script/e2e.sh b/script/e2e.sh index 71b05ae..8c89f82 100644 --- a/script/e2e.sh +++ b/script/e2e.sh @@ -10,10 +10,8 @@ response_token=$(curl -s --location "$BASE_URL$TOKEN_ENDPOINT" \ --header 'Content-Type: application/json' \ --data '{"username": "user", "password": "password"}') -# Extract the token from the response using jq (ensure jq is installed) token=$(echo "$response_token" | jq -r '.token') -# Check if the token was successfully retrieved if [ -z "$token" ]; then echo "Error: Failed to retrieve token." exit 1 @@ -24,23 +22,13 @@ echo "----------------------------------" # Step 2: Create 3 chat sessions for the user and get the first session's ID for i in {1..3}; do - # Make a POST request to create a chat session response=$(curl -s --location --request POST "$BASE_URL/users/$USER_ID/chat-sessions" \ --header "Authorization: Bearer $token") - # Extract the chat session ID from the response of the first chat session if [ $i -eq 1 ]; then CHAT_SESSION_ID=$(echo "$response" | jq -r '.id') echo "First chat session created with ID: $CHAT_SESSION_ID" fi - - # Check if the session creation was successful - # if [[ $(echo "$response" | jq -r '.status') == "success" ]]; then - # echo "Chat session $i created successfully." - # else - # echo "Error creating chat session $i." - # exit - # fi done # Step 3: Send the first message @@ -57,7 +45,7 @@ response_message_2=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/ --header "Authorization: Bearer $token" \ --data "{\"content\":\"$message_content_2\"}") -# Step 4.5: Send the third message +# Step 5: Send the third message message_content_3="what social media do they use the most" response_message_3=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ --header "Content-Type: application/json" \ @@ -66,25 +54,15 @@ response_message_3=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/ system_message_id_3=$(echo "$response_message_3" | jq -r '.systemMessage.id') -## Step 5: Display the responses from sending the messages -#echo "Message 1 sent to chat session $CHAT_SESSION_ID:" -#echo "$response_message_1" -# -#echo "Message 2 sent to chat session $CHAT_SESSION_ID:" -#echo "$response_message_2" -# -#echo "Message 3 sent to chat session $CHAT_SESSION_ID:" -#echo "$response_message_3" - # Step 6: Fetch and display the whole chat session echo "Fetching the entire chat session $CHAT_SESSION_ID..." chat_session_response=$(curl -s --location "$BASE_URL/chat-sessions/$CHAT_SESSION_ID" \ --header "Authorization: Bearer $token") -# Display the full chat session echo "Full Chat Session $CHAT_SESSION_ID:" echo "$chat_session_response" +# Step 7: Submit feedback for the third message feedback_message="The force is not strong with that message." feedback_response=$(curl -s --location --write-out "%{http_code}" --request POST "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages/$system_message_id_3/feedback" \ --header "Authorization: Bearer $token" \ @@ -96,3 +74,14 @@ else echo "Error: Feedback submission failed. Status code: $feedback_response" exit 1 fi + +# Step 8: Send the final and failed message +message_content_4="what are butterflies" +response_message_4=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ + --header "Content-Type: application/json" \ + --header "Authorization: Bearer $token" \ + --data "{\"content\":\"$message_content_4\"}") + + +echo "Message 4 that's supposed to be not answered sent to chat session $CHAT_SESSION_ID:" +echo "$response_message_4" From 0ef324fae15a0f3d432e5e8838c855ca635c73c0 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 23:48:15 +0300 Subject: [PATCH 29/52] Swagger definition --- docs/swagger/docs.go | 44 +++++++++++-------- docs/swagger/swagger.json | 44 +++++++++++-------- docs/swagger/swagger.yaml | 38 ++++++++-------- .../chatSessions/submitFeedbackHandler.go | 20 ++++----- 4 files changed, 82 insertions(+), 64 deletions(-) diff --git a/docs/swagger/docs.go b/docs/swagger/docs.go index fbecc66..ed29f8e 100644 --- a/docs/swagger/docs.go +++ b/docs/swagger/docs.go @@ -167,28 +167,21 @@ const docTemplate = `{ "BearerAuth": [] } ], - "description": "Sends message to a given chat session and gets response", - "summary": "Sends message to a given chat session and gets response", + "description": "Submits a feedback to a message", + "summary": "Submits a feedback to a message", "parameters": [ { "description": "request body", - "name": "SendMessageRequest", + "name": "SubmitFeedbackRequest", "in": "body", "required": true, "schema": { - "$ref": "#/definitions/chatSessions.SendMessageRequest" + "$ref": "#/definitions/chatSessions.SubmitFeedbackRequest" } }, { - "type": "integer", - "description": "user id", - "name": "user_id", - "in": "path", - "required": true - }, - { - "description": "session id", - "name": "session_id", + "description": "message_id", + "name": "message_id", "in": "body", "required": true, "schema": { @@ -200,25 +193,25 @@ const docTemplate = `{ "201": { "description": "Created", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } }, "400": { "description": "Error in message payload", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } }, "401": { "description": "Authentication error", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } }, "500": { "description": "Internal Server Error", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } } } @@ -276,7 +269,6 @@ const docTemplate = `{ "type": "object", "properties": { "content": { - "description": "UserID string ` + "`" + `json:\"chatSessionID\"` + "`" + `\nSessionID string ` + "`" + `json:\"sessionId\"` + "`" + `", "type": "string" } } @@ -295,6 +287,22 @@ const docTemplate = `{ } } }, + "chatSessions.SubmitFeedbackRequest": { + "type": "object", + "properties": { + "feedback": { + "type": "string" + } + } + }, + "chatSessions.SubmitFeedbackResponse": { + "type": "object", + "properties": { + "errorMessage": { + "type": "string" + } + } + }, "chatSessions.UserChatSessionsResponse": { "type": "object", "properties": { diff --git a/docs/swagger/swagger.json b/docs/swagger/swagger.json index 4d844be..7fcc076 100644 --- a/docs/swagger/swagger.json +++ b/docs/swagger/swagger.json @@ -161,28 +161,21 @@ "BearerAuth": [] } ], - "description": "Sends message to a given chat session and gets response", - "summary": "Sends message to a given chat session and gets response", + "description": "Submits a feedback to a message", + "summary": "Submits a feedback to a message", "parameters": [ { "description": "request body", - "name": "SendMessageRequest", + "name": "SubmitFeedbackRequest", "in": "body", "required": true, "schema": { - "$ref": "#/definitions/chatSessions.SendMessageRequest" + "$ref": "#/definitions/chatSessions.SubmitFeedbackRequest" } }, { - "type": "integer", - "description": "user id", - "name": "user_id", - "in": "path", - "required": true - }, - { - "description": "session id", - "name": "session_id", + "description": "message_id", + "name": "message_id", "in": "body", "required": true, "schema": { @@ -194,25 +187,25 @@ "201": { "description": "Created", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } }, "400": { "description": "Error in message payload", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } }, "401": { "description": "Authentication error", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } }, "500": { "description": "Internal Server Error", "schema": { - "$ref": "#/definitions/chatSessions.SendMessageResponse" + "$ref": "#/definitions/chatSessions.SubmitFeedbackResponse" } } } @@ -270,7 +263,6 @@ "type": "object", "properties": { "content": { - "description": "UserID string `json:\"chatSessionID\"`\nSessionID string `json:\"sessionId\"`", "type": "string" } } @@ -289,6 +281,22 @@ } } }, + "chatSessions.SubmitFeedbackRequest": { + "type": "object", + "properties": { + "feedback": { + "type": "string" + } + } + }, + "chatSessions.SubmitFeedbackResponse": { + "type": "object", + "properties": { + "errorMessage": { + "type": "string" + } + } + }, "chatSessions.UserChatSessionsResponse": { "type": "object", "properties": { diff --git a/docs/swagger/swagger.yaml b/docs/swagger/swagger.yaml index ef446fb..b226326 100644 --- a/docs/swagger/swagger.yaml +++ b/docs/swagger/swagger.yaml @@ -35,9 +35,6 @@ definitions: chatSessions.SendMessageRequest: properties: content: - description: |- - UserID string `json:"chatSessionID"` - SessionID string `json:"sessionId"` type: string type: object chatSessions.SendMessageResponse: @@ -49,6 +46,16 @@ definitions: userMessage: $ref: '#/definitions/chatSessions.MessageResponse' type: object + chatSessions.SubmitFeedbackRequest: + properties: + feedback: + type: string + type: object + chatSessions.SubmitFeedbackResponse: + properties: + errorMessage: + type: string + type: object chatSessions.UserChatSessionsResponse: properties: errorMessage: @@ -155,22 +162,17 @@ paths: summary: Creates chat session /users/user_id/chat-sessions/session_id/messages: post: - description: Sends message to a given chat session and gets response + description: Submits a feedback to a message parameters: - description: request body in: body - name: SendMessageRequest + name: SubmitFeedbackRequest required: true schema: - $ref: '#/definitions/chatSessions.SendMessageRequest' - - description: user id - in: path - name: user_id - required: true - type: integer - - description: session id + $ref: '#/definitions/chatSessions.SubmitFeedbackRequest' + - description: message_id in: body - name: session_id + name: message_id required: true schema: type: integer @@ -178,22 +180,22 @@ paths: "201": description: Created schema: - $ref: '#/definitions/chatSessions.SendMessageResponse' + $ref: '#/definitions/chatSessions.SubmitFeedbackResponse' "400": description: Error in message payload schema: - $ref: '#/definitions/chatSessions.SendMessageResponse' + $ref: '#/definitions/chatSessions.SubmitFeedbackResponse' "401": description: Authentication error schema: - $ref: '#/definitions/chatSessions.SendMessageResponse' + $ref: '#/definitions/chatSessions.SubmitFeedbackResponse' "500": description: Internal Server Error schema: - $ref: '#/definitions/chatSessions.SendMessageResponse' + $ref: '#/definitions/chatSessions.SubmitFeedbackResponse' security: - BearerAuth: [] - summary: Sends message to a given chat session and gets response + summary: Submits a feedback to a message produces: - application/json securityDefinitions: diff --git a/internal/handlers/chatSessions/submitFeedbackHandler.go b/internal/handlers/chatSessions/submitFeedbackHandler.go index 9345ffa..b5f97d0 100644 --- a/internal/handlers/chatSessions/submitFeedbackHandler.go +++ b/internal/handlers/chatSessions/submitFeedbackHandler.go @@ -25,16 +25,16 @@ func NewSubmitFeedbackHandler( } } -// @Summary Submits a feedback to a message -// @Description Submits a feedback to a message -// @Security BearerAuth -// @Param SubmitFeedbackRequest body SubmitFeedbackRequest true "request body" -// @Param message_id body int true "message_id" -// @Success 201 {object} SubmitFeedbackResponse -// @Failure 400 {object} SubmitFeedbackResponse "Error in message payload" -// @Failure 401 {object} SubmitFeedbackResponse "Authentication error" -// @Failure 500 {object} SubmitFeedbackResponse "Internal Server Error" -// @Router /users/user_id/chat-sessions/session_id/messages [post] +// @Summary Submits a feedback to a message +// @Description Submits a feedback to a message +// @Security BearerAuth +// @Param SubmitFeedbackRequest body SubmitFeedbackRequest true "request body" +// @Param message_id body int true "message_id" +// @Success 201 {object} SubmitFeedbackResponse +// @Failure 400 {object} SubmitFeedbackResponse "Error in message payload" +// @Failure 401 {object} SubmitFeedbackResponse "Authentication error" +// @Failure 500 {object} SubmitFeedbackResponse "Internal Server Error" +// @Router /users/user_id/chat-sessions/session_id/messages [post] func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") From 388aaf1c7fe460e3fe115f8998707978f336b39b Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Tue, 27 May 2025 23:48:28 +0300 Subject: [PATCH 30/52] README finish vol1 --- README.md | 99 +++++++++++++++++------------------- examples/submitFeedback.http | 18 +++++++ script/Makefile | 52 +------------------ 3 files changed, 68 insertions(+), 101 deletions(-) create mode 100644 examples/submitFeedback.http diff --git a/README.md b/README.md index 93a0ae8..c38f342 100644 --- a/README.md +++ b/README.md @@ -1,11 +1,11 @@ -# Platform Go Challenge +# Jedi Team Challenge - Louk Chatwalker --- ## Description -Service that provides a REST API offering CRUD operations for adding, removing, editing -and getting a User's favourite Assets (Chart, Insight, Audience) +Service that provides a REST API offering creating chat sessions, sending messages, reading the chat session and +submitting feedback for a message. --- @@ -13,40 +13,40 @@ and getting a User's favourite Assets (Chart, Insight, Audience) `cd script && make start-app` -* This command will start the app with `localhost` address and `:8080` port (specified in build/Dockerfile.dev and .env) +* This command will start the app with `localhost` address and `:8080` port (specified in build/Dev.Dockerfile and .env) -`cd script && make migrate-data-small` +Then you can create, get User's chat-sessions, send message and get response from the Knowledge Base (data.md) +like the examples in /examples directory. To generate the needed Bearer token, please call /token endpoint with +username = "user" +& password = "password" like in the example. -* This command will add to DB 10 Users and 3 Assets per User for testing reasons +This runs the app with "dlv" so that we can also attach a debugger while running. -`cd script && make migrate-data-large` +Also you can run the /scripts/e2e.sh script to run all cases of the assignment: -* This command will add to DB 10 Users and 10.000 Assets per User for testing reasons - -Then you can add, edit, remove and get User's favourites like the examples in /examples -directory. To generate the needed Bearer token, please call /token endpoint with username -& password like in the example +1. It creates a JWT token +2. It creates three chat sessions for that User +3. In the first chat session we send three messages related to each other, so that the history + is shown: + 1. "what do you know about latino mobile gamers?" + 2. "do they use social media?" + 3. "what social media do they use the most?" +4. It shows the whole chat session that shows the whole story +5. It submits a negative feedback for the last message +6. It sends a final message that the chat is not supposed to answer (what are butterflies) --- ## Makefile Commands -| Command | Usage | -|---------------------------------|------------------------------------------------------------------------| -| start-app | `Start app` | -| kill-app | `Stop app` | -| rebuild-app | `Rebuild app` | -| tests-all | `Run both unit and integration tests` | -| tests-benchmark | `Run benchmark tests` | -| tests-unit | `Run unit tests ` | -| tests-file FILE={filePath} | `Run specific file test` | -| generate-mock FILE={filePath} | `Generate mock for a specific file` | -| run-linter | `Runs linter` | -| generate-swagger-files | `Generates swagger.json definitions in Docs dir` | -| tests-package PACKAGE={package} | `Run specific package test` | -| tests-all-with-coverage | `Run both unit and integration tests via docker with coverage details` | -| migrate-data-small | `Run migration of a small dataset Users & Assets in the DB` | -| migrate-data-large | `Run migration of a larger dataset Users & Assets in the DB` | +| Command | Usage | +|-------------------------------|--------------------------------------------------| +| start-app | `Start app` | +| kill-app | `Stop app` | +| rebuild-app | `Rebuild app in case of code changes` | +| tests | `Run both unit and integration tests` | +| generate-mock FILE={filePath} | `Generate mock for a specific file` | +| swag | `Generates swagger.json definitions in Docs dir` | * All these are executed through docker containers * In order to execute makefile commands type **make** plus a command from the table above @@ -57,36 +57,33 @@ directory. To generate the needed Bearer token, please call /token endpoint with ## Notes -1. .env is pushed to Git only for the Assessment purpose. Config and .env files should never be tracked. +1. .env is not pushed to Git, so in order to run the app you need it with secret keys (e.g Pinecone, OpenAI etc) 2. There are three Dockerfile files. 1. Dockerfile is the normal, production one - 2. Dockerfile.dev is for setting up a remote debugger Delve - 3. Dockerfile.utilities is for building a docker for "utilities" like running tests, + 2. Dev.Dockerfile is for setting up a remote debugger Delve + 3. Utilities.Dockerfile is for building a docker for "utilities" like running tests, linting etc -3. Asset types' fields and their validation was done very simply for the scope of this assessment -4. As described at the assignment, the User sees a list of Assets and adds some to their - favorites (by starring them for example). So the Assets are existing. This is why - the AddAsset() takes just IDs and just associates a User with an Asset, and not create it. +4. LLM Choices I took: (This is a field that it's a bit unknown for me, so some decision were made with just a little + studying + and might not be the correct ones): + 1. Pinecone for vector database + 2. text-embedding-3-small as embedding model + 3. Tittoken as a tokenizer with CHUNK_ENCODING_MODEL cl100k_base and MAX_TOKENS_PER_CHUNKS 3000 +5. There are swagger definitions in /docs, and examples in /examples that show the usage of the API. And the e2e.sh that + checks everything. ## Known Issues -1. Unit tests for adding, removing, editing assets are done only for Charts and only for happy paths. +1. Only happy path tests are created due to time constraints. 2. JWT mechanism just requires a fake username and password to generate a JWT token and does NOT do - actual login due to lack of time. Also no test created for it - -## Performance - -If Users favourites data were causing performance issues, I would investigate the following solutions: - -1. Add a cache to the repository layer, over the DB and keep there the User's favourite Assets - in order to have faster access to it -1. Limit Payload by implementing Pagination -1. Limit Payload by compressing the data with Accept-Encoding: gzip or compress -1. Add indexes in DB. For example we could add indexed to the Users-Assets tables e.g "users_charts" on user_id - and chart_id - -* All these solutions would need to be discussed with the Engineering team to find the most suitable as - they all have their pros and cons + actual login due to lack of time. Also no test created for it. Also, the user_id that exists in the endpoint should + come + the JWT directly. +3. In my implementation, when inputing the Chat History from the Messages DB, I import all messages to OpenAI so that + the + discussion gets continued. In a production env, I would not do that, but put a limit to the number of messages read + from + history, as there might be a lot of messages. ## Security diff --git a/examples/submitFeedback.http b/examples/submitFeedback.http new file mode 100644 index 0000000..9014133 --- /dev/null +++ b/examples/submitFeedback.http @@ -0,0 +1,18 @@ +### + +# curl --location 'localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions/8d010ba8-5761-42ec-9442-7b6fcc19280e/messages/c4dca9d2-3c73-4356-897f-75791d43ca39' +#--header 'Content-Type: application/json' +#--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4Mzc4NjgzLCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.Nf7TCuFTbyqapG3IIiN9j9f4oMVI1ANv14aP9BV44l4' +#--data '{ +# "feedback":"abla" +#}' +POST localhost:8080/users/12345678-0000-0000-0000-000000000000/chat-sessions/8d010ba8-5761-42ec-9442-7b6fcc19280e/messages/c4dca9d2-3c73-4356-897f-75791d43ca39 +Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ1c2VyIiwiZXhwIjoxNzQ4Mzc4NjgzLCJVc2VySW5mbyI6eyJVc2VybmFtZSI6InVzZXIiLCJQYXNzd29yZCI6InBhc3N3b3JkIn19.Nf7TCuFTbyqapG3IIiN9j9f4oMVI1ANv14aP9BV44l4 +Content-Type: application/json + +{ + "feedback": "abla" +} + +### + diff --git a/script/Makefile b/script/Makefile index bfc0ace..4dee214 100644 --- a/script/Makefile +++ b/script/Makefile @@ -7,23 +7,7 @@ endif @docker run --volume "$(PWD)"/../:/app --workdir /app \ dev-build /bin/bash -c "mockgen -source=${FILE} -destination=mocks/mock_${FILE}" -tests-unit: - make build-dev - @docker run \ - --rm \ - --volume "$(PWD)"/../:/app \ - --workdir /app \ - dev-build go test -short -cover -count=1 ./... - -tests-benchmark: - make build-dev - @docker run \ - --rm \ - --volume "$(PWD)"/../:/app \ - --workdir /app \ - dev-build go test ./... -bench=. - -tests-all: +tests: make build-dev @docker run \ --rm \ @@ -31,22 +15,6 @@ tests-all: --workdir /app \ dev-build go test ./... -cover -count=1 -tests-all-with-coverage: - make build-dev - @docker run \ - --rm \ - --volume "$(PWD)"/../:/app \ - --workdir /app \ - dev-build go test -count=1 -v -coverpkg=./... -coverprofile=profile.cov ./... ; go tool cover -func profile.cov - -run-linter: - make build-dev - @docker run \ - --rm \ - --volume "$(PWD)"/../:/app \ - --workdir /app \ - dev-build golangci-lint run --config ./config/.golangci.yml ./... - swag-fmt: # swag fmt -g ../../pkg/server/routes.go -d ../internal/handlers make build-dev @@ -80,20 +48,4 @@ rebuild-app: build-dev: @docker build \ --tag dev-build \ - -f ../build/Utilities.Dockerfile .. - -migrate-data-small: - @docker run \ - --rm \ - --volume "$(PWD)":/scripts \ - -e PGPASSWORD=password \ - postgres \ - psql -h 172.18.0.1 -p 5432 -U username -d mydb -w -f ./scripts/migrationFileSmall.sql - -migrate-data-large: - @docker run \ - --rm \ - --volume "$(PWD)":/scripts \ - -e PGPASSWORD=password \ - postgres \ - psql -h 172.18.0.1 -p 5432 -U username -d mydb -w -f ./scripts/migrationFileLarge.sql \ No newline at end of file + -f ../build/Utilities.Dockerfile .. \ No newline at end of file From b575962e15f099383358a23d9977b5183a537458 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 00:08:19 +0300 Subject: [PATCH 31/52] Adds some README info --- README.md | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index c38f342..c1d7db6 100644 --- a/README.md +++ b/README.md @@ -68,7 +68,11 @@ Also you can run the /scripts/e2e.sh script to run all cases of the assignment: and might not be the correct ones): 1. Pinecone for vector database 2. text-embedding-3-small as embedding model - 3. Tittoken as a tokenizer with CHUNK_ENCODING_MODEL cl100k_base and MAX_TOKENS_PER_CHUNKS 3000 + 3. Tiktoken as a tokenizer with CHUNK_ENCODING_MODEL cl100k_base and MAX_TOKENS_PER_CHUNKS 3000 + 4. The top 7 results are retrieved from the similarity search in the Vector DB, and there is a threshold of 0.35 + that rejects the matches with score less than that. If no such matches are found, then the answer is "The force + is not strong enough for me to answer that question based on my context." + 5. For OpenAI model I have chosen "gpt-4.1-nano" which is a nice combination and balance of speed, accuracy and price. 5. There are swagger definitions in /docs, and examples in /examples that show the usage of the API. And the e2e.sh that checks everything. From 7d94fa82035e22e4ecfe55fe84602c1b31bd60b6 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 00:13:15 +0300 Subject: [PATCH 32/52] Small fixes --- internal/core/services/messageService.go | 13 +++- .../chatSessions/submitFeedbackHandler.go | 75 ++++++++++++++++--- .../submitFeedbackHandler_test.go | 15 ++-- .../core/services/messageService.go | 8 +- 4 files changed, 86 insertions(+), 25 deletions(-) diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index f2eef95..c1d3892 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -18,7 +18,7 @@ import ( type MessageServiceInterface interface { CreateMessage(context.Context, uuid.UUID, *domain.Message) (uuid.UUID, error) GetAnswerForMessage(context.Context, uuid.UUID) (*domain.Message, error) - UpdateMessageFeedback(context.Context, uuid.UUID, string) error + UpdateMessageFeedback(ctx context.Context, message *domain.Message, userID uuid.UUID) error } type Embedder interface { @@ -120,9 +120,14 @@ func (s *MessageService) GetAnswerForMessage(ctx context.Context, initialMessage accumulatedTextFromSearch, err := s.vectorDB.SemanticSearch(ctx, vectorToFloat32) - answer, err := s.generateAnswerFromOpenAI(ctx, accumulatedTextFromSearch, initialMessage.Content, chatSession.Messages) - if err != nil { - return nil, err + var answer string + if len(accumulatedTextFromSearch) == 0 { + answer = "The force is not strong enough for me to answer that question based on my context." + } else { + answer, err = s.generateAnswerFromOpenAI(ctx, accumulatedTextFromSearch, initialMessage.Content, chatSession.Messages) + if err != nil { + return nil, err + } } replyMessage := &domain.Message{ diff --git a/internal/handlers/chatSessions/submitFeedbackHandler.go b/internal/handlers/chatSessions/submitFeedbackHandler.go index b5f97d0..1b87130 100644 --- a/internal/handlers/chatSessions/submitFeedbackHandler.go +++ b/internal/handlers/chatSessions/submitFeedbackHandler.go @@ -4,6 +4,7 @@ import ( "encoding/json" "github.com/google/uuid" "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" "github.com/loukaspe/jedi-team-challenge/internal/core/services" customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" "github.com/loukaspe/jedi-team-challenge/pkg/logger" @@ -25,16 +26,16 @@ func NewSubmitFeedbackHandler( } } -// @Summary Submits a feedback to a message -// @Description Submits a feedback to a message -// @Security BearerAuth -// @Param SubmitFeedbackRequest body SubmitFeedbackRequest true "request body" -// @Param message_id body int true "message_id" -// @Success 201 {object} SubmitFeedbackResponse -// @Failure 400 {object} SubmitFeedbackResponse "Error in message payload" -// @Failure 401 {object} SubmitFeedbackResponse "Authentication error" -// @Failure 500 {object} SubmitFeedbackResponse "Internal Server Error" -// @Router /users/user_id/chat-sessions/session_id/messages [post] +// @Summary Submits a feedback to a message +// @Description Submits a feedback to a message +// @Security BearerAuth +// @Param SubmitFeedbackRequest body SubmitFeedbackRequest true "request body" +// @Param message_id body int true "message_id" +// @Success 201 {object} SubmitFeedbackResponse +// @Failure 400 {object} SubmitFeedbackResponse "Error in message payload" +// @Failure 401 {object} SubmitFeedbackResponse "Authentication error" +// @Failure 500 {object} SubmitFeedbackResponse "Internal Server Error" +// @Router /users/user_id/chat-sessions/session_id/messages [post] func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") @@ -67,6 +68,52 @@ func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWr return } + userIDAsString := mux.Vars(r)["user_id"] + if userIDAsString == "" { + response.ErrorMessage = "missing user id" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + userID, err := uuid.Parse(userIDAsString) + if err != nil { + handler.logger.Error("Error in submitting feedback", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed message uuid" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + chatSessionIDAsString := mux.Vars(r)["session_id"] + if chatSessionIDAsString == "" { + response.ErrorMessage = "missing session id" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + + chatSessionID, err := uuid.Parse(chatSessionIDAsString) + if err != nil { + handler.logger.Error("Error in submitting feedback", + map[string]interface{}{ + "errorMessage": err.Error(), + }) + + response.ErrorMessage = "malformed sessio uuid" + + handler.JsonResponse(w, http.StatusBadRequest, response) + + return + } + err = json.NewDecoder(r.Body).Decode(request) if err != nil { handler.logger.Error("Error in submitting feedback", @@ -83,8 +130,12 @@ func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWr err = handler.MessageService.UpdateMessageFeedback( ctx, - messageID, - request.Feedback, + &domain.Message{ + ID: messageID, + Feedback: &request.Feedback, + ChatSessionID: chatSessionID, + }, + userID, ) if resourceNotFound, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { diff --git a/internal/handlers/chatSessions/submitFeedbackHandler_test.go b/internal/handlers/chatSessions/submitFeedbackHandler_test.go index 04adf36..036dc18 100644 --- a/internal/handlers/chatSessions/submitFeedbackHandler_test.go +++ b/internal/handlers/chatSessions/submitFeedbackHandler_test.go @@ -6,6 +6,7 @@ import ( "encoding/json" "github.com/google/uuid" "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" mock_services "github.com/loukaspe/jedi-team-challenge/mocks/mock_internal/core/services" "github.com/loukaspe/jedi-team-challenge/pkg/logger" "github.com/stretchr/testify/assert" @@ -57,9 +58,9 @@ func TestUpdateMessageFeedbackHandler_UpdateMessageFeedbackController(t *testing ) vars := map[string]string{ - "user_id": tt.args.userID.String(), - "chat_session_id": tt.args.chatSessionID.String(), - "message_id": tt.args.messageID.String(), + "user_id": tt.args.userID.String(), + "session_id": tt.args.chatSessionID.String(), + "message_id": tt.args.messageID.String(), } mockRequest = mux.SetURLVars(mockRequest, vars) @@ -68,8 +69,12 @@ func TestUpdateMessageFeedbackHandler_UpdateMessageFeedbackController(t *testing mockService.EXPECT().UpdateMessageFeedback( gomock.Any(), - tt.args.messageID, - tt.args.feedback, + &domain.Message{ + ID: tt.args.messageID, + Feedback: &tt.args.feedback, + ChatSessionID: tt.args.chatSessionID, + }, + tt.args.userID, ).Return(nil) handler := &SubmitFeedbackHandler{ diff --git a/mocks/mock_internal/core/services/messageService.go b/mocks/mock_internal/core/services/messageService.go index 036e584..ee1243c 100644 --- a/mocks/mock_internal/core/services/messageService.go +++ b/mocks/mock_internal/core/services/messageService.go @@ -72,17 +72,17 @@ func (mr *MockMessageServiceInterfaceMockRecorder) GetAnswerForMessage(arg0, arg } // UpdateMessageFeedback mocks base method. -func (m *MockMessageServiceInterface) UpdateMessageFeedback(arg0 context.Context, arg1 uuid.UUID, arg2 string) error { +func (m *MockMessageServiceInterface) UpdateMessageFeedback(ctx context.Context, message *domain.Message, userID uuid.UUID) error { m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "UpdateMessageFeedback", arg0, arg1, arg2) + ret := m.ctrl.Call(m, "UpdateMessageFeedback", ctx, message, userID) ret0, _ := ret[0].(error) return ret0 } // UpdateMessageFeedback indicates an expected call of UpdateMessageFeedback. -func (mr *MockMessageServiceInterfaceMockRecorder) UpdateMessageFeedback(arg0, arg1, arg2 any) *gomock.Call { +func (mr *MockMessageServiceInterfaceMockRecorder) UpdateMessageFeedback(ctx, message, userID any) *gomock.Call { mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "UpdateMessageFeedback", reflect.TypeOf((*MockMessageServiceInterface)(nil).UpdateMessageFeedback), arg0, arg1, arg2) + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "UpdateMessageFeedback", reflect.TypeOf((*MockMessageServiceInterface)(nil).UpdateMessageFeedback), ctx, message, userID) } // MockEmbedder is a mock of Embedder interface. From 10b4c410c66cdea713f5a6e89a16a3fca61393c7 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 09:58:14 +0300 Subject: [PATCH 33/52] Make script messages more clear --- script/e2e.sh | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/script/e2e.sh b/script/e2e.sh index 8c89f82..a46b093 100644 --- a/script/e2e.sh +++ b/script/e2e.sh @@ -31,6 +31,8 @@ for i in {1..3}; do fi done +echo "----------------------------------" + # Step 3: Send the first message message_content_1="what do you know about latino mobile gamers" response_message_1=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ @@ -54,6 +56,9 @@ response_message_3=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/ system_message_id_3=$(echo "$response_message_3" | jq -r '.systemMessage.id') +echo "Three messages were sent in the same discussion and title was generated" +echo "----------------------------------" + # Step 6: Fetch and display the whole chat session echo "Fetching the entire chat session $CHAT_SESSION_ID..." chat_session_response=$(curl -s --location "$BASE_URL/chat-sessions/$CHAT_SESSION_ID" \ @@ -62,6 +67,8 @@ chat_session_response=$(curl -s --location "$BASE_URL/chat-sessions/$CHAT_SESSIO echo "Full Chat Session $CHAT_SESSION_ID:" echo "$chat_session_response" +echo "----------------------------------" + # Step 7: Submit feedback for the third message feedback_message="The force is not strong with that message." feedback_response=$(curl -s --location --write-out "%{http_code}" --request POST "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages/$system_message_id_3/feedback" \ @@ -69,12 +76,14 @@ feedback_response=$(curl -s --location --write-out "%{http_code}" --request POST --data "{\"feedback\":\"$feedback_message\"}") if [ "$feedback_response" -eq 201 ]; then - echo "Feedback OK" + echo "Negative feedback OK: received 201 status code" else echo "Error: Feedback submission failed. Status code: $feedback_response" exit 1 fi +echo "----------------------------------" + # Step 8: Send the final and failed message message_content_4="what are butterflies" response_message_4=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages" \ From 8db959b27444b97c79671930ad6343de4aaf769a Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 10:10:14 +0300 Subject: [PATCH 34/52] Fix test --- internal/repositories/messageRepositoryCreate_test.go | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/internal/repositories/messageRepositoryCreate_test.go b/internal/repositories/messageRepositoryCreate_test.go index e15fdb3..e95f999 100644 --- a/internal/repositories/messageRepositoryCreate_test.go +++ b/internal/repositories/messageRepositoryCreate_test.go @@ -40,9 +40,10 @@ func TestChatRepository_CreateMessage(t *testing.T) { Sender: "USER", Content: "ablaabla", CreatedAt: time.Time{}, + Feedback: nil, }, }, - mockSqlMessageQueryExpected: `INSERT INTO "messages" ("chat_session_id","sender","content","created_at") VALUES ($1,$2,$3,$4) RETURNING "id"`, + mockSqlMessageQueryExpected: `INSERT INTO "messages" ("chat_session_id","sender","content","created_at","feedback") VALUES ($1,$2,$3,$4,$5) RETURNING "id"`, mockInsertedMessageIdReturned: uuid.UUID{0x42, 0x34, 0x56, 0x78}, expectedMessageUid: uuid.UUID{0x42, 0x34, 0x56, 0x78}, }, @@ -58,7 +59,7 @@ func TestChatRepository_CreateMessage(t *testing.T) { mockDb.ExpectQuery(regexp.QuoteMeta(tt.mockSqlMessageQueryExpected)). WithArgs( - tt.args.message.ChatSessionID, tt.args.message.Sender, tt.args.message.Content, sqlmock.AnyArg(), + tt.args.message.ChatSessionID, tt.args.message.Sender, tt.args.message.Content, sqlmock.AnyArg(), tt.args.message.Feedback, ). WillReturnRows(sqlmock.NewRows([]string{"id"}).AddRow(tt.mockInsertedMessageIdReturned)) mockDb.ExpectCommit() From e4e159cde80b12b1a7d936a0d1116a07e3398b68 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 10:10:23 +0300 Subject: [PATCH 35/52] Minor readme fix --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index c1d7db6..230ea38 100644 --- a/README.md +++ b/README.md @@ -67,8 +67,8 @@ Also you can run the /scripts/e2e.sh script to run all cases of the assignment: studying and might not be the correct ones): 1. Pinecone for vector database - 2. text-embedding-3-small as embedding model - 3. Tiktoken as a tokenizer with CHUNK_ENCODING_MODEL cl100k_base and MAX_TOKENS_PER_CHUNKS 3000 + 2. "text-embedding-3-small" as embedding model + 3. Tiktoken as a tokenizer with CHUNK_ENCODING_MODEL "cl100k_base" and MAX_TOKENS_PER_CHUNKS 3000 4. The top 7 results are retrieved from the similarity search in the Vector DB, and there is a threshold of 0.35 that rejects the matches with score less than that. If no such matches are found, then the answer is "The force is not strong enough for me to answer that question based on my context." From 0844fd6ce1d49fb920e80a7d427899c0e971dfac Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 16:43:25 +0300 Subject: [PATCH 36/52] Remove TODOs --- internal/handlers/chatSessions/submitFeedbackHandler_test.go | 1 - internal/repositories/messageRepositoryUpdateFeedback.go | 1 - 2 files changed, 2 deletions(-) diff --git a/internal/handlers/chatSessions/submitFeedbackHandler_test.go b/internal/handlers/chatSessions/submitFeedbackHandler_test.go index 036dc18..90d32cd 100644 --- a/internal/handlers/chatSessions/submitFeedbackHandler_test.go +++ b/internal/handlers/chatSessions/submitFeedbackHandler_test.go @@ -46,7 +46,6 @@ func TestUpdateMessageFeedbackHandler_UpdateMessageFeedbackController(t *testing }, } - //TODO: user mismatch sto feedback for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { mockRequest := httptest.NewRequest( diff --git a/internal/repositories/messageRepositoryUpdateFeedback.go b/internal/repositories/messageRepositoryUpdateFeedback.go index 9f21729..3391c95 100644 --- a/internal/repositories/messageRepositoryUpdateFeedback.go +++ b/internal/repositories/messageRepositoryUpdateFeedback.go @@ -21,7 +21,6 @@ func (repo *MessageRepository) UpdateMessageFeedback( Update("feedback", feedback).Error if err == gorm.ErrRecordNotFound { - // TODO: customerrors return customerrors.ResourceNotFoundErrorWrapper{ OriginalError: errors.New("messageID " + uuid.String() + " not found"), } From 43e8606a6804b25357c8ec0df35d47a106d50120 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 17:03:06 +0300 Subject: [PATCH 37/52] Improve readme --- README.md | 50 ++++++++++++++++++++++++++++---------------------- 1 file changed, 28 insertions(+), 22 deletions(-) diff --git a/README.md b/README.md index 230ea38..643b254 100644 --- a/README.md +++ b/README.md @@ -4,8 +4,9 @@ ## Description -Service that provides a REST API offering creating chat sessions, sending messages, reading the chat session and -submitting feedback for a message. +This service provides a REST API that enables the creation of chat sessions, sending messages, reading chat sessions, +and submitting feedback for messages. The responses to the chat messages are generated using a Retrieval-Augmented +Generation (RAG) approach, leveraging a provided dataset of GWI data. --- @@ -16,13 +17,12 @@ submitting feedback for a message. * This command will start the app with `localhost` address and `:8080` port (specified in build/Dev.Dockerfile and .env) Then you can create, get User's chat-sessions, send message and get response from the Knowledge Base (data.md) -like the examples in /examples directory. To generate the needed Bearer token, please call /token endpoint with -username = "user" -& password = "password" like in the example. +like the examples in `/examples` directory. To generate the needed Bearer token, please call `/token` endpoint with +username = "user" & password = "password" like in the example. This runs the app with "dlv" so that we can also attach a debugger while running. -Also you can run the /scripts/e2e.sh script to run all cases of the assignment: +Also you can run the `sh /scripts/e2e.sh` script to run all cases of the assignment: 1. It creates a JWT token 2. It creates three chat sessions for that User @@ -61,34 +61,40 @@ Also you can run the /scripts/e2e.sh script to run all cases of the assignment: 2. There are three Dockerfile files. 1. Dockerfile is the normal, production one 2. Dev.Dockerfile is for setting up a remote debugger Delve - 3. Utilities.Dockerfile is for building a docker for "utilities" like running tests, - linting etc -4. LLM Choices I took: (This is a field that it's a bit unknown for me, so some decision were made with just a little - studying - and might not be the correct ones): + 3. Utilities.Dockerfile is for building a docker for "utilities" like running tests, linting etc +4. LLM Choices (Made with limited knowledge): 1. Pinecone for vector database - 2. "text-embedding-3-small" as embedding model - 3. Tiktoken as a tokenizer with CHUNK_ENCODING_MODEL "cl100k_base" and MAX_TOKENS_PER_CHUNKS 3000 + 2. `text-embedding-3-small` as embedding model + 3. Tiktoken as a tokenizer with CHUNK_ENCODING_MODEL `cl100k_base` and MAX_TOKENS_PER_CHUNKS `3000` 4. The top 7 results are retrieved from the similarity search in the Vector DB, and there is a threshold of 0.35 that rejects the matches with score less than that. If no such matches are found, then the answer is "The force is not strong enough for me to answer that question based on my context." - 5. For OpenAI model I have chosen "gpt-4.1-nano" which is a nice combination and balance of speed, accuracy and price. -5. There are swagger definitions in /docs, and examples in /examples that show the usage of the API. And the e2e.sh that + 5. For OpenAI model I have chosen `gpt-4.1-nano` which is a nice combination and balance of speed, accuracy and price. +5. There are swagger definitions in `/docs`, and examples in `/examples` that show the usage of the API. And the `e2e.sh` that checks everything. +6. My approach for the code structure is the Hexagonal Architecture, more on that https://medium.com/@matiasvarela/hexagonal-architecture-in-go-cfd4e436faa3 ## Known Issues 1. Only happy path tests are created due to time constraints. 2. JWT mechanism just requires a fake username and password to generate a JWT token and does NOT do actual login due to lack of time. Also no test created for it. Also, the user_id that exists in the endpoint should - come - the JWT directly. + come the JWT directly. 3. In my implementation, when inputing the Chat History from the Messages DB, I import all messages to OpenAI so that - the - discussion gets continued. In a production env, I would not do that, but put a limit to the number of messages read - from - history, as there might be a lot of messages. + the discussion gets continued. In a production env, I would not do that, but put a limit to the number of messages read + from history, as there might be a lot of messages. ## Security -1. JWT mechanism added for Authentication and Authorization (incomplete - see Known Issues) \ No newline at end of file +1. JWT mechanism added for Authentication and Authorization (incomplete - see Known Issues) + +## Libraries and Tools + +1. github.com/gorilla/mux for routing +2. gorm.io/gorm as ORM for my PostgreSQL DB +3. github.com/golang-jwt/jwt/v4 for the JWT token handling +4. github.com/openai/openai-go for communicating with OpenAI +5. github.com/pinecone-io/go-pinecone/v3 for Pinecone Vector DB +6. github.com/pkoukk/tiktoken-go for tokenizer +7. github.com/swaggo/swag for Swagger definition +8. github.com/stretchr/testify & go.uber.org/mock & github.com/DATA-DOG/go-sqlmock for testing \ No newline at end of file From 9f88592deb0995764bf3fffd1f13ae8017ec6f1a Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 17:17:12 +0300 Subject: [PATCH 38/52] Small readme addition --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 643b254..4791d95 100644 --- a/README.md +++ b/README.md @@ -83,6 +83,8 @@ Also you can run the `sh /scripts/e2e.sh` script to run all cases of the assignm 3. In my implementation, when inputing the Chat History from the Messages DB, I import all messages to OpenAI so that the discussion gets continued. In a production env, I would not do that, but put a limit to the number of messages read from history, as there might be a lot of messages. +4. For performance increase, we can put indices in the DB, on the foreign keys so that the fetch in the GET + endpoints is faster. ## Security From f9d84e087e1d3b7af1ae609bc052cf6a94ff9a1a Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 17:23:33 +0300 Subject: [PATCH 39/52] Swagger fix --- docs/swagger/docs.go | 64 +++++++++++++++++++++++++++++++++++++++ docs/swagger/swagger.json | 64 +++++++++++++++++++++++++++++++++++++++ docs/swagger/swagger.yaml | 41 +++++++++++++++++++++++++ 3 files changed, 169 insertions(+) diff --git a/docs/swagger/docs.go b/docs/swagger/docs.go index ed29f8e..b03c152 100644 --- a/docs/swagger/docs.go +++ b/docs/swagger/docs.go @@ -161,6 +161,70 @@ const docTemplate = `{ } }, "/users/user_id/chat-sessions/session_id/messages": { + "post": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Sends message to a given chat session and gets response", + "summary": "Sends message to a given chat session and gets response", + "parameters": [ + { + "description": "request body", + "name": "SendMessageRequest", + "in": "body", + "required": true, + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageRequest" + } + }, + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + }, + { + "description": "session id", + "name": "session_id", + "in": "body", + "required": true, + "schema": { + "type": "integer" + } + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + } + } + } + }, + "/users/user_id/chat-sessions/session_id/messages/message_id/feedback": { "post": { "security": [ { diff --git a/docs/swagger/swagger.json b/docs/swagger/swagger.json index 7fcc076..136578c 100644 --- a/docs/swagger/swagger.json +++ b/docs/swagger/swagger.json @@ -155,6 +155,70 @@ } }, "/users/user_id/chat-sessions/session_id/messages": { + "post": { + "security": [ + { + "BearerAuth": [] + } + ], + "description": "Sends message to a given chat session and gets response", + "summary": "Sends message to a given chat session and gets response", + "parameters": [ + { + "description": "request body", + "name": "SendMessageRequest", + "in": "body", + "required": true, + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageRequest" + } + }, + { + "type": "integer", + "description": "user id", + "name": "user_id", + "in": "path", + "required": true + }, + { + "description": "session id", + "name": "session_id", + "in": "body", + "required": true, + "schema": { + "type": "integer" + } + } + ], + "responses": { + "201": { + "description": "Created", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "400": { + "description": "Error in message payload", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "401": { + "description": "Authentication error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + }, + "500": { + "description": "Internal Server Error", + "schema": { + "$ref": "#/definitions/chatSessions.SendMessageResponse" + } + } + } + } + }, + "/users/user_id/chat-sessions/session_id/messages/message_id/feedback": { "post": { "security": [ { diff --git a/docs/swagger/swagger.yaml b/docs/swagger/swagger.yaml index b226326..f102b5f 100644 --- a/docs/swagger/swagger.yaml +++ b/docs/swagger/swagger.yaml @@ -161,6 +161,47 @@ paths: - BearerAuth: [] summary: Creates chat session /users/user_id/chat-sessions/session_id/messages: + post: + description: Sends message to a given chat session and gets response + parameters: + - description: request body + in: body + name: SendMessageRequest + required: true + schema: + $ref: '#/definitions/chatSessions.SendMessageRequest' + - description: user id + in: path + name: user_id + required: true + type: integer + - description: session id + in: body + name: session_id + required: true + schema: + type: integer + responses: + "201": + description: Created + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + "400": + description: Error in message payload + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + "401": + description: Authentication error + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + "500": + description: Internal Server Error + schema: + $ref: '#/definitions/chatSessions.SendMessageResponse' + security: + - BearerAuth: [] + summary: Sends message to a given chat session and gets response + /users/user_id/chat-sessions/session_id/messages/message_id/feedback: post: description: Submits a feedback to a message parameters: From c2d76c59f987829ae82dcda93adaf898fccb2730 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 17:23:42 +0300 Subject: [PATCH 40/52] Swagger fix vol2 --- internal/handlers/chatSessions/submitFeedbackHandler.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/internal/handlers/chatSessions/submitFeedbackHandler.go b/internal/handlers/chatSessions/submitFeedbackHandler.go index 1b87130..279a1f2 100644 --- a/internal/handlers/chatSessions/submitFeedbackHandler.go +++ b/internal/handlers/chatSessions/submitFeedbackHandler.go @@ -35,7 +35,7 @@ func NewSubmitFeedbackHandler( // @Failure 400 {object} SubmitFeedbackResponse "Error in message payload" // @Failure 401 {object} SubmitFeedbackResponse "Authentication error" // @Failure 500 {object} SubmitFeedbackResponse "Internal Server Error" -// @Router /users/user_id/chat-sessions/session_id/messages [post] +// @Router /users/user_id/chat-sessions/session_id/messages/message_id/feedback [post] func (handler *SubmitFeedbackHandler) SubmitFeedbackController(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") From 06ee169bf9a64b3e60f62a92941fde14f211e7d3 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 17:25:39 +0300 Subject: [PATCH 41/52] small readme fix --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 4791d95..8061bca 100644 --- a/README.md +++ b/README.md @@ -57,7 +57,7 @@ Also you can run the `sh /scripts/e2e.sh` script to run all cases of the assignm ## Notes -1. .env is not pushed to Git, so in order to run the app you need it with secret keys (e.g Pinecone, OpenAI etc) +1. `/config/.env` and `deployment/.env` are not pushed to Git, so in order to run the app you need them with secret keys (e.g Pinecone, OpenAI etc) 2. There are three Dockerfile files. 1. Dockerfile is the normal, production one 2. Dev.Dockerfile is for setting up a remote debugger Delve From 835f4a5d043c5a9e0b3dec70fa050b797d26f47a Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 28 May 2025 17:47:07 +0300 Subject: [PATCH 42/52] Minor fixes --- README.md | 1 + pkg/chunks/chunker.go | 4 +--- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 8061bca..008962f 100644 --- a/README.md +++ b/README.md @@ -70,6 +70,7 @@ Also you can run the `sh /scripts/e2e.sh` script to run all cases of the assignm that rejects the matches with score less than that. If no such matches are found, then the answer is "The force is not strong enough for me to answer that question based on my context." 5. For OpenAI model I have chosen `gpt-4.1-nano` which is a nice combination and balance of speed, accuracy and price. + 6. I've put a rate limiter when calling OpenAI because at times I was having 429 Many Request response. 5. There are swagger definitions in `/docs`, and examples in `/examples` that show the usage of the API. And the `e2e.sh` that checks everything. 6. My approach for the code structure is the Hexagonal Architecture, more on that https://medium.com/@matiasvarela/hexagonal-architecture-in-go-cfd4e436faa3 diff --git a/pkg/chunks/chunker.go b/pkg/chunks/chunker.go index 5f95409..f2166b3 100644 --- a/pkg/chunks/chunker.go +++ b/pkg/chunks/chunker.go @@ -10,8 +10,6 @@ type Chunker struct { MaxTokensPerChunk int } -// TiktokenEncoding string -// enc, err := tiktoken.GetEncoding(encName) func NewChunker(encoder *tiktoken.Tiktoken, maxTokens int) (*Chunker, error) { return &Chunker{Encoder: encoder, MaxTokensPerChunk: maxTokens}, nil } @@ -66,7 +64,7 @@ func splitToSentences(text string) []string { sb.Reset() } } - // leftover + if rem := strings.TrimSpace(sb.String()); rem != "" { sents = append(sents, rem) } From 14571e3b799e34f7f50384afcdb201b816de8a65 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Thu, 29 May 2025 13:16:40 +0300 Subject: [PATCH 43/52] Add example response in README --- README.md | 50 +++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 49 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 008962f..01a3a56 100644 --- a/README.md +++ b/README.md @@ -100,4 +100,52 @@ Also you can run the `sh /scripts/e2e.sh` script to run all cases of the assignm 5. github.com/pinecone-io/go-pinecone/v3 for Pinecone Vector DB 6. github.com/pkoukk/tiktoken-go for tokenizer 7. github.com/swaggo/swag for Swagger definition -8. github.com/stretchr/testify & go.uber.org/mock & github.com/DATA-DOG/go-sqlmock for testing \ No newline at end of file +8. github.com/stretchr/testify & go.uber.org/mock & github.com/DATA-DOG/go-sqlmock for testing + +## Example Chat Session Response + +``` +{ + "id": "5488a398-1801-4a7c-ba6d-69d833453313", + "title": "Latino Mobile Gamers Overview", + "createdAt": "2025-05-29 10:13:58.368579 +0000 UTC", + "updatedAt": "2025-05-29 10:13:59.831385 +0000 UTC", + "messages": [ + { + "id": "7a93a664-17fa-40a4-bd59-7d2c81d7accf", + "sender": "USER", + "content": "what do you know about latino mobile gamers", + "created_at": "2025-05-29 10:13:58.417981 +0000 UTC" + }, + { + "id": "3bf684be-39ad-4fd0-8cae-27d42ce56111", + "sender": "SYSTEM", + "content": "Based on the provided context, Latino mobile gamers are characterized by the following behaviors and interests:\n- They are 49% more likely to visit Reddit daily compared to the average person.\n- They are 51% more likely to use TikTok weekly compared to the average person.\n- They are 42% more likely to use TikTok daily compared to the average person.\n- They are 36% more likely to use Instagram more than once a day than the average person.\n- They are 74% more likely to be interested in Esports compared to the average person.\n- They are 49% more likely to buy products or services to access the community built around them.\n- They are 62% more likely to find out about new brands and products through vlogs.\n- They are 43% more likely to discover new brands through ads seen in video or mobile games.\n- They are 62% more likely to discover new brands and products through posts or reviews from expert bloggers.\n- They are 25% between the ages of 16 and 24, and 22% between 25 and 34.\n- They are 30% more likely to be between 25 and 34 years old than the average person.\n- They are 16% of U.S. Hispanic/Latino mobile gamers.\n- They are 42% more likely to be interested in computers and coding.\n- They are 74% more likely to discover new brands and products through vlogs.\n- They are 103% more likely to discover new brands through ads in video or mobile games compared to the average person.\n- They are 45% more likely to spend 2-3 hours on streaming services daily.\n- They are 41% more likely to spend more than 4 hours on streaming services daily.\nThis indicates that Latino mobile gamers are highly active on social media platforms like TikTok and Instagram, have a strong interest in gaming, esports, technology, and discovering new brands through video content and ads in mobile games.", + "created_at": "2025-05-29 10:14:11.136485 +0000 UTC" + }, + { + "id": "86718efc-d4b7-46b6-ac3f-b8c1a044cd8d", + "sender": "USER", + "content": "do they use social media", + "created_at": "2025-05-29 10:14:11.172935 +0000 UTC" + }, + { + "id": "4dbba2a9-8df5-4c29-97f9-dd7d8999a34e", + "sender": "SYSTEM", + "content": "Yes, based on the provided context, Latino mobile gamers actively use social media. They are more likely than the average person to use platforms such as TikTok (42% more likely to use weekly, 36% more likely to use daily), Instagram (62% more likely to use more than once a day), and Reddit (49% more likely to visit daily). They also frequently discover new brands and products through social media content like vlogs and reviews from bloggers, indicating high engagement with social media channels.", + "created_at": "2025-05-29 10:14:17.505033 +0000 UTC" + }, + { + "id": "e63bbd20-0519-45bd-a88f-48e0d4179c9e", + "sender": "USER", + "content": "what social media do they use the most", + "created_at": "2025-05-29 10:14:17.541732 +0000 UTC" + }, + { + "id": "28e0d999-19b5-40d7-a23c-b31012fa8c87", + "sender": "SYSTEM", + "content": "Based on the provided context, Latino mobile gamers use TikTok and Instagram the most. They are 42% more likely to use TikTok weekly, 36% more likely to use it more than once a day, and 62% more likely to use Instagram more than once a day compared to the average person.", + "created_at": "2025-05-29 10:14:23.200825 +0000 UTC" + } + ] +}``` \ No newline at end of file From 75fe317f6888316ccc9c4fd2b51a9549fd3cbfdf Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 18 Jun 2025 12:42:14 +0300 Subject: [PATCH 44/52] Add option to input another source --- cmd/http/main.go | 41 +- data.md => dataPeople.md | 0 dataVehicles.md | 1002 ++++++++++++++++++++++++++++++++++++++ pkg/vectordb/pinecone.go | 11 +- 4 files changed, 1049 insertions(+), 5 deletions(-) rename data.md => dataPeople.md (100%) create mode 100644 dataVehicles.md diff --git a/cmd/http/main.go b/cmd/http/main.go index 104ac92..cdb3752 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -34,7 +34,7 @@ func main() { embedder := getEmbedder(&client) pineconeVectorDB := getPineconeVectorDB() - inputKnowledgeBase(ctx, chunker, embedder, pineconeVectorDB) + inputPeopleKnowledgeBase(ctx, chunker, embedder, pineconeVectorDB) logger := logger.NewLogger(ctx) router := mux.NewRouter() @@ -174,8 +174,8 @@ func getPineconeVectorDB() *vectordb.PineconeVectorDB { } -func inputKnowledgeBase(ctx context.Context, chunker *chunks.Chunker, embedder *embeddings.EmbeddingService, pineconeVectorDB *vectordb.PineconeVectorDB) { - textBytes, err := os.ReadFile("./data.md") +func inputVehiclesKnowledgeBase(ctx context.Context, chunker *chunks.Chunker, embedder *embeddings.EmbeddingService, pineconeVectorDB *vectordb.PineconeVectorDB) { + textBytes, err := os.ReadFile("./dataVehicles.md") if err != nil { log.Fatal(err) } @@ -189,7 +189,40 @@ func inputKnowledgeBase(ctx context.Context, chunker *chunks.Chunker, embedder * log.Fatalf("Embedding error: %v", err) } - count, err := pineconeVectorDB.StoreEmbeddings(ctx, domainEmbeddings) + count, err := pineconeVectorDB.StoreEmbeddings( + ctx, + domainEmbeddings, + map[string]interface{}{ + "type": "vehicles", + }) + if err != nil { + log.Fatalf("Failed to store embeddings: %v", err) + } + + fmt.Sprintf("Stored %d embeddings in Pinecone index %s\n", count, os.Getenv("PINECONE_INDEX")) +} + +func inputPeopleKnowledgeBase(ctx context.Context, chunker *chunks.Chunker, embedder *embeddings.EmbeddingService, pineconeVectorDB *vectordb.PineconeVectorDB) { + textBytes, err := os.ReadFile("./dataPeople.md") + if err != nil { + log.Fatal(err) + } + text := string(textBytes) + + chunks := chunker.Chunk(text) + fmt.Printf("Generated %d chunks\n", len(chunks)) + + domainEmbeddings, err := embedder.Embed(ctx, chunks) + if err != nil { + log.Fatalf("Embedding error: %v", err) + } + + count, err := pineconeVectorDB.StoreEmbeddings( + ctx, + domainEmbeddings, + map[string]interface{}{ + "type": "people", + }) if err != nil { log.Fatalf("Failed to store embeddings: %v", err) } diff --git a/data.md b/dataPeople.md similarity index 100% rename from data.md rename to dataPeople.md diff --git a/dataVehicles.md b/dataVehicles.md new file mode 100644 index 0000000..5c4a209 --- /dev/null +++ b/dataVehicles.md @@ -0,0 +1,1002 @@ +| text | +|:----------------------------------------------------------------------------| +| Pickup trucks are 46% efficient than Scooters. | +| Sailboats are 88% slow than Electric cars. | +| Commercial airplanes are 89% expensive than Sports cars. | +| Speedboats are 172% quick to brake than Helicopters. | +| SUVs are 117% cost-effective than Private jets. | +| Sedans are 101% affordable than Pickup trucks. | +| Electric cars are 152% quick to accelerate than Hybrid vehicles. | +| Private jets are 18% versatile than Road bikes. | +| Electric cars are 176% powerful than Road bikes. | +| Sedans are 26% environmentally friendly than Speedboats. | +| Sedans are 106% heavy-duty than Sailboats. | +| Scooters are 130% fast than Mountain bikes. | +| SUVs are 154% environmentally friendly than Speedboats. | +| Cargo ships are 160% environmentally friendly than Electric cars. | +| Sailboats are 51% cost-effective than Road bikes. | +| Commercial airplanes are 29% quick to brake than SUVs. | +| Cargo ships are 143% affordable than Hybrid vehicles. | +| Scooters are 193% fast than Sailboats. | +| Submarines are 171% lightweight than Scooters. | +| Sports cars are 160% lightweight than Pickup trucks. | +| Helicopters are 23% expensive than Electric cars. | +| Helicopters are 166% quick to brake than Commercial airplanes. | +| Sailboats are 111% affordable than Scooters. | +| Electric bikes are 24% quick to brake than Sailboats. | +| Submarines are 123% cost-effective than Sports cars. | +| Electric cars are 85% lightweight than Private jets. | +| Pickup trucks are 144% versatile than Mountain bikes. | +| Commercial airplanes are 126% quick to accelerate than Motorcycles. | +| Cargo ships are 99% affordable than Sedans. | +| Mountain bikes are 171% durable than Road bikes. | +| Sports cars are 60% slow than Trains. | +| Hybrid vehicles are 7% affordable than Motorcycles. | +| Electric bikes are 175% environmentally friendly than Pickup trucks. | +| Cargo ships are 106% maneuverable than Private jets. | +| Yachts are 161% powerful than Motorcycles. | +| Yachts are 131% quiet than Electric cars. | +| Private jets are 6% affordable than Helicopters. | +| Sedans are 42% durable than Mountain bikes. | +| Sports cars are 182% durable than Commercial airplanes. | +| Sailboats are 59% cost-effective than Speedboats. | +| Pickup trucks are 66% quick to accelerate than Sports cars. | +| Commercial airplanes are 64% stable than Electric bikes. | +| Cargo ships are 133% affordable than Sailboats. | +| Pickup trucks are 12% durable than Motorcycles. | +| Sports cars are 39% durable than Trains. | +| Submarines are 41% heavy-duty than Sports cars. | +| Sports cars are 99% stable than Hybrid vehicles. | +| Sedans are 19% environmentally friendly than Scooters. | +| Pickup trucks are 33% maneuverable than Sports cars. | +| Pickup trucks are 141% lightweight than Road bikes. | +| Submarines are 1% quick to brake than Scooters. | +| Speedboats are 2% durable than Scooters. | +| Mountain bikes are 4% versatile than Sports cars. | +| Electric cars are 43% cost-effective than Trains. | +| Motorcycles are 190% quiet than Pickup trucks. | +| Electric cars are 84% efficient than Commercial airplanes. | +| Electric bikes are 93% versatile than Speedboats. | +| Road bikes are 52% powerful than Hybrid vehicles. | +| Road bikes are 30% quiet than Electric bikes. | +| Mountain bikes are 118% expensive than Road bikes. | +| Speedboats are 198% fuel-efficient than Cargo ships. | +| Electric bikes are 110% heavy-duty than Submarines. | +| Sailboats are 178% noisy than SUVs. | +| Cargo ships are 127% slow than Yachts. | +| Motorcycles are 10% stable than Mountain bikes. | +| Yachts are 9% cost-effective than Scooters. | +| Mountain bikes are 200% quiet than Cargo ships. | +| Helicopters are 95% stable than Scooters. | +| Commercial airplanes are 16% lightweight than Scooters. | +| Sailboats are 62% lightweight than Scooters. | +| Submarines are 69% stable than Yachts. | +| Pickup trucks are 55% quick to brake than Submarines. | +| Private jets are 149% quick to accelerate than Yachts. | +| Hybrid vehicles are 52% cost-effective than Helicopters. | +| Yachts are 113% quick to brake than Hybrid vehicles. | +| Submarines are 129% quiet than Mountain bikes. | +| Sports cars are 176% cost-effective than Scooters. | +| Mountain bikes are 26% versatile than Private jets. | +| Helicopters are 72% efficient than Hybrid vehicles. | +| Hybrid vehicles are 191% stable than SUVs. | +| Private jets are 124% environmentally friendly than Motorcycles. | +| Electric bikes are 175% expensive than SUVs. | +| Electric bikes are 113% expensive than Sports cars. | +| Sedans are 45% slow than Electric bikes. | +| Scooters are 64% efficient than Hybrid vehicles. | +| Sports cars are 93% lightweight than Speedboats. | +| Scooters are 168% powerful than Motorcycles. | +| Speedboats are 31% quick to accelerate than Sailboats. | +| Sedans are 56% efficient than Pickup trucks. | +| Yachts are 122% heavy-duty than Hybrid vehicles. | +| Pickup trucks are 182% environmentally friendly than Cargo ships. | +| Cargo ships are 181% maneuverable than Motorcycles. | +| Trains are 144% lightweight than Motorcycles. | +| Mountain bikes are 172% versatile than Electric bikes. | +| Yachts are 153% cost-effective than SUVs. | +| Submarines are 166% powerful than Hybrid vehicles. | +| SUVs are 52% environmentally friendly than Electric cars. | +| Helicopters are 93% maneuverable than Scooters. | +| Hybrid vehicles are 12% quick to accelerate than Submarines. | +| Trains are 116% noisy than Sedans. | +| Pickup trucks are 46% slow than Scooters. | +| Yachts are 107% quick to brake than Sailboats. | +| Sedans are 4% quick to brake than Private jets. | +| Mountain bikes are 59% cost-effective than Helicopters. | +| Motorcycles are 42% fast than Electric cars. | +| Mountain bikes are 2% maneuverable than SUVs. | +| Sedans are 14% cost-effective than SUVs. | +| Motorcycles are 113% quick to accelerate than Hybrid vehicles. | +| Submarines are 134% cost-effective than Hybrid vehicles. | +| Hybrid vehicles are 188% efficient than Electric cars. | +| Private jets are 98% quick to brake than Electric bikes. | +| Yachts are 73% fast than Mountain bikes. | +| SUVs are 74% lightweight than Yachts. | +| Submarines are 167% lightweight than Speedboats. | +| Road bikes are 178% slow than Sports cars. | +| Helicopters are 141% noisy than Electric cars. | +| Private jets are 199% fast than Sailboats. | +| Road bikes are 172% quick to accelerate than Motorcycles. | +| Commercial airplanes are 49% environmentally friendly than Hybrid vehicles. | +| SUVs are 152% environmentally friendly than Electric cars. | +| Commercial airplanes are 1% powerful than Road bikes. | +| Hybrid vehicles are 118% efficient than Electric cars. | +| Private jets are 194% quick to accelerate than Pickup trucks. | +| Hybrid vehicles are 39% stable than SUVs. | +| Electric bikes are 161% affordable than Motorcycles. | +| Yachts are 28% quick to brake than Commercial airplanes. | +| Cargo ships are 60% affordable than Yachts. | +| Cargo ships are 128% slow than Trains. | +| Cargo ships are 57% affordable than Private jets. | +| Trains are 169% environmentally friendly than Pickup trucks. | +| Sedans are 21% quiet than Cargo ships. | +| Yachts are 94% quiet than Submarines. | +| Helicopters are 45% environmentally friendly than Sports cars. | +| Mountain bikes are 69% environmentally friendly than Trains. | +| Sedans are 58% durable than Sailboats. | +| Submarines are 11% efficient than Yachts. | +| Pickup trucks are 40% expensive than Sedans. | +| Motorcycles are 18% cost-effective than Private jets. | +| Sedans are 173% quiet than Sailboats. | +| Helicopters are 80% noisy than Sports cars. | +| Sailboats are 23% maneuverable than Electric bikes. | +| Scooters are 33% expensive than Pickup trucks. | +| Yachts are 24% durable than Sports cars. | +| Private jets are 47% powerful than Trains. | +| Road bikes are 154% expensive than Electric bikes. | +| Scooters are 84% versatile than Electric bikes. | +| Electric cars are 178% quick to accelerate than Sports cars. | +| Road bikes are 173% fuel-efficient than Yachts. | +| Motorcycles are 49% powerful than Sailboats. | +| Yachts are 64% quick to brake than Submarines. | +| Motorcycles are 170% lightweight than Submarines. | +| Mountain bikes are 125% powerful than Sports cars. | +| Cargo ships are 96% heavy-duty than Private jets. | +| Electric cars are 20% durable than Hybrid vehicles. | +| Private jets are 6% stable than SUVs. | +| SUVs are 93% environmentally friendly than Hybrid vehicles. | +| Scooters are 110% heavy-duty than Hybrid vehicles. | +| Sedans are 180% stable than Speedboats. | +| SUVs are 115% quick to brake than Sailboats. | +| Sedans are 71% stable than Yachts. | +| Road bikes are 44% expensive than Electric bikes. | +| Cargo ships are 170% fuel-efficient than Private jets. | +| Submarines are 81% quiet than Yachts. | +| Pickup trucks are 59% versatile than Scooters. | +| Road bikes are 9% fuel-efficient than Helicopters. | +| Sports cars are 140% stable than Trains. | +| Yachts are 78% lightweight than Submarines. | +| Sports cars are 196% quick to accelerate than SUVs. | +| Electric cars are 82% durable than Cargo ships. | +| Electric cars are 92% durable than Scooters. | +| Road bikes are 95% cost-effective than Speedboats. | +| Electric cars are 192% environmentally friendly than Scooters. | +| Commercial airplanes are 118% fast than Road bikes. | +| Submarines are 119% heavy-duty than Commercial airplanes. | +| Cargo ships are 95% quick to accelerate than Sedans. | +| Trains are 75% fast than Mountain bikes. | +| Hybrid vehicles are 1% stable than Road bikes. | +| Commercial airplanes are 41% fuel-efficient than Scooters. | +| Hybrid vehicles are 83% expensive than Submarines. | +| Speedboats are 113% environmentally friendly than Yachts. | +| Submarines are 186% fast than Pickup trucks. | +| Sailboats are 92% versatile than Motorcycles. | +| Helicopters are 190% durable than Private jets. | +| Pickup trucks are 18% versatile than Road bikes. | +| Cargo ships are 46% cost-effective than Private jets. | +| Yachts are 76% slow than Sailboats. | +| Sailboats are 72% expensive than Yachts. | +| Sedans are 26% cost-effective than Electric cars. | +| Electric cars are 196% noisy than Pickup trucks. | +| Private jets are 60% fuel-efficient than Commercial airplanes. | +| Road bikes are 141% quiet than Electric bikes. | +| Commercial airplanes are 40% expensive than Yachts. | +| Sports cars are 38% affordable than Mountain bikes. | +| Speedboats are 28% fuel-efficient than Commercial airplanes. | +| Private jets are 85% fuel-efficient than Electric bikes. | +| Road bikes are 116% fast than Hybrid vehicles. | +| Sedans are 169% efficient than Sailboats. | +| Mountain bikes are 28% expensive than SUVs. | +| Electric bikes are 63% affordable than Yachts. | +| Hybrid vehicles are 123% efficient than Speedboats. | +| Road bikes are 14% efficient than Scooters. | +| Motorcycles are 73% fast than Electric bikes. | +| Motorcycles are 154% cost-effective than Submarines. | +| SUVs are 166% quiet than Sports cars. | +| Helicopters are 153% durable than Private jets. | +| Sailboats are 77% fast than Helicopters. | +| Sports cars are 132% fuel-efficient than Electric bikes. | +| Sports cars are 189% fuel-efficient than Sailboats. | +| Yachts are 46% noisy than Hybrid vehicles. | +| Sailboats are 76% stable than Motorcycles. | +| Private jets are 54% fuel-efficient than Submarines. | +| Speedboats are 53% versatile than Sedans. | +| Motorcycles are 66% quick to accelerate than Trains. | +| Road bikes are 172% quick to accelerate than Sports cars. | +| Pickup trucks are 172% affordable than Scooters. | +| Scooters are 152% efficient than Helicopters. | +| Hybrid vehicles are 21% expensive than Commercial airplanes. | +| Road bikes are 18% affordable than Electric bikes. | +| Trains are 96% durable than Submarines. | +| Helicopters are 173% powerful than Sports cars. | +| Submarines are 114% maneuverable than Helicopters. | +| Helicopters are 160% quick to accelerate than Sedans. | +| Hybrid vehicles are 199% slow than Helicopters. | +| SUVs are 105% affordable than Sports cars. | +| Motorcycles are 108% durable than Road bikes. | +| Motorcycles are 55% heavy-duty than Trains. | +| Cargo ships are 180% noisy than Private jets. | +| Motorcycles are 52% powerful than Trains. | +| Submarines are 131% heavy-duty than Hybrid vehicles. | +| SUVs are 107% affordable than Road bikes. | +| Electric bikes are 27% fast than Road bikes. | +| Submarines are 64% expensive than Electric cars. | +| Hybrid vehicles are 61% expensive than Speedboats. | +| Commercial airplanes are 42% cost-effective than Mountain bikes. | +| Road bikes are 140% durable than Commercial airplanes. | +| Submarines are 99% versatile than Private jets. | +| Hybrid vehicles are 144% durable than Helicopters. | +| Pickup trucks are 136% versatile than Submarines. | +| Private jets are 39% durable than Mountain bikes. | +| Road bikes are 153% quick to accelerate than Electric cars. | +| Road bikes are 40% expensive than Cargo ships. | +| Commercial airplanes are 62% quiet than Private jets. | +| Road bikes are 74% stable than Commercial airplanes. | +| Electric bikes are 51% quick to brake than Motorcycles. | +| Yachts are 1% slow than Mountain bikes. | +| Private jets are 153% powerful than Mountain bikes. | +| Mountain bikes are 164% quiet than Pickup trucks. | +| Trains are 114% quick to brake than Helicopters. | +| Scooters are 100% powerful than Submarines. | +| Pickup trucks are 173% environmentally friendly than Helicopters. | +| Trains are 189% quick to accelerate than Cargo ships. | +| Pickup trucks are 43% quiet than Helicopters. | +| Sailboats are 168% fast than Trains. | +| Trains are 45% stable than Electric cars. | +| Trains are 89% powerful than Cargo ships. | +| Hybrid vehicles are 17% powerful than Speedboats. | +| Sports cars are 92% quick to accelerate than Motorcycles. | +| Mountain bikes are 195% maneuverable than Road bikes. | +| Hybrid vehicles are 31% slow than Pickup trucks. | +| Mountain bikes are 56% durable than Electric bikes. | +| Yachts are 192% powerful than Mountain bikes. | +| Motorcycles are 29% quick to brake than Sedans. | +| Electric bikes are 152% lightweight than Private jets. | +| Submarines are 100% affordable than Sedans. | +| Speedboats are 192% environmentally friendly than Yachts. | +| Yachts are 104% heavy-duty than Scooters. | +| Commercial airplanes are 8% heavy-duty than Sedans. | +| Cargo ships are 146% stable than Sedans. | +| Speedboats are 24% quiet than Scooters. | +| Electric bikes are 180% quiet than Sailboats. | +| Cargo ships are 2% environmentally friendly than Mountain bikes. | +| Trains are 155% durable than Sports cars. | +| Motorcycles are 110% quiet than Speedboats. | +| Sports cars are 116% quiet than Electric bikes. | +| Private jets are 48% fuel-efficient than Sedans. | +| Private jets are 74% fuel-efficient than Scooters. | +| Mountain bikes are 3% heavy-duty than Scooters. | +| Sailboats are 111% cost-effective than Pickup trucks. | +| Speedboats are 12% noisy than Yachts. | +| Private jets are 67% slow than Helicopters. | +| Electric cars are 168% lightweight than Trains. | +| Yachts are 176% lightweight than Electric cars. | +| Submarines are 11% expensive than Yachts. | +| Sailboats are 43% versatile than Road bikes. | +| Yachts are 155% noisy than SUVs. | +| Commercial airplanes are 110% maneuverable than Electric bikes. | +| Submarines are 22% slow than Road bikes. | +| Hybrid vehicles are 34% quiet than Road bikes. | +| Electric bikes are 69% powerful than Submarines. | +| Helicopters are 112% expensive than Commercial airplanes. | +| Mountain bikes are 45% noisy than Scooters. | +| Scooters are 86% quick to brake than Motorcycles. | +| Sports cars are 75% durable than Electric bikes. | +| Yachts are 113% maneuverable than Cargo ships. | +| Motorcycles are 1% cost-effective than Sailboats. | +| Sailboats are 99% slow than Cargo ships. | +| Sailboats are 193% lightweight than Scooters. | +| Helicopters are 104% versatile than Submarines. | +| Speedboats are 133% cost-effective than Sports cars. | +| Commercial airplanes are 186% maneuverable than Motorcycles. | +| Pickup trucks are 44% heavy-duty than Sports cars. | +| Helicopters are 165% cost-effective than Motorcycles. | +| Submarines are 98% quiet than Mountain bikes. | +| Scooters are 87% fuel-efficient than Speedboats. | +| Trains are 24% heavy-duty than Sailboats. | +| Submarines are 126% durable than Sedans. | +| SUVs are 68% cost-effective than Hybrid vehicles. | +| Electric bikes are 137% lightweight than SUVs. | +| Helicopters are 147% durable than Sedans. | +| Electric bikes are 24% quiet than Sedans. | +| Scooters are 48% maneuverable than Commercial airplanes. | +| Motorcycles are 23% powerful than Yachts. | +| Trains are 124% cost-effective than Yachts. | +| Yachts are 23% heavy-duty than Private jets. | +| Helicopters are 87% efficient than Pickup trucks. | +| Sailboats are 184% fuel-efficient than Hybrid vehicles. | +| Road bikes are 152% versatile than Mountain bikes. | +| Mountain bikes are 29% affordable than SUVs. | +| Sedans are 11% versatile than Cargo ships. | +| Scooters are 14% lightweight than Motorcycles. | +| Private jets are 176% efficient than Submarines. | +| Submarines are 39% affordable than Electric bikes. | +| Mountain bikes are 153% efficient than Hybrid vehicles. | +| Sedans are 183% stable than Pickup trucks. | +| Cargo ships are 197% efficient than Helicopters. | +| Scooters are 128% quick to accelerate than Sedans. | +| Yachts are 71% efficient than Electric cars. | +| Mountain bikes are 197% environmentally friendly than Private jets. | +| Commercial airplanes are 90% durable than Helicopters. | +| Motorcycles are 100% affordable than Cargo ships. | +| Cargo ships are 191% versatile than Electric bikes. | +| Cargo ships are 110% fuel-efficient than Road bikes. | +| Hybrid vehicles are 163% expensive than Helicopters. | +| Trains are 44% versatile than Speedboats. | +| Road bikes are 24% heavy-duty than Pickup trucks. | +| Trains are 154% stable than Electric cars. | +| Pickup trucks are 66% expensive than Electric cars. | +| Scooters are 37% powerful than Motorcycles. | +| Yachts are 57% affordable than Helicopters. | +| Submarines are 95% fast than Hybrid vehicles. | +| Sedans are 35% stable than Commercial airplanes. | +| Electric cars are 196% fuel-efficient than Sports cars. | +| Sports cars are 164% stable than Trains. | +| Submarines are 153% quick to accelerate than Cargo ships. | +| SUVs are 152% heavy-duty than Sports cars. | +| Sedans are 193% noisy than SUVs. | +| Sedans are 49% versatile than Speedboats. | +| Mountain bikes are 49% expensive than Pickup trucks. | +| Electric cars are 89% powerful than Sailboats. | +| Commercial airplanes are 104% expensive than SUVs. | +| Electric bikes are 85% durable than Sports cars. | +| Private jets are 137% fast than Submarines. | +| Sports cars are 101% durable than Cargo ships. | +| Sports cars are 57% heavy-duty than Pickup trucks. | +| Pickup trucks are 147% quiet than Road bikes. | +| Pickup trucks are 108% fast than Sports cars. | +| Private jets are 47% maneuverable than SUVs. | +| Hybrid vehicles are 117% cost-effective than Sailboats. | +| Sports cars are 59% quiet than Pickup trucks. | +| Helicopters are 88% efficient than Sports cars. | +| Motorcycles are 199% stable than Sports cars. | +| Mountain bikes are 170% lightweight than Scooters. | +| Hybrid vehicles are 160% quick to accelerate than Sedans. | +| SUVs are 158% affordable than Trains. | +| Helicopters are 1% quick to accelerate than Electric cars. | +| Sedans are 103% environmentally friendly than Hybrid vehicles. | +| Mountain bikes are 188% powerful than Yachts. | +| Commercial airplanes are 38% heavy-duty than Submarines. | +| Electric bikes are 95% efficient than Motorcycles. | +| Sailboats are 28% powerful than Road bikes. | +| Sedans are 37% lightweight than Sailboats. | +| Yachts are 53% durable than Sailboats. | +| Private jets are 83% quick to accelerate than Commercial airplanes. | +| Mountain bikes are 112% fast than Road bikes. | +| Submarines are 198% quick to accelerate than Scooters. | +| Commercial airplanes are 40% efficient than Sports cars. | +| Helicopters are 89% quiet than Sports cars. | +| Road bikes are 70% expensive than Yachts. | +| Trains are 196% affordable than Speedboats. | +| Cargo ships are 91% fuel-efficient than Road bikes. | +| Helicopters are 168% versatile than Hybrid vehicles. | +| SUVs are 87% powerful than Private jets. | +| Speedboats are 73% quick to brake than Yachts. | +| Helicopters are 161% quick to brake than Pickup trucks. | +| Sports cars are 7% slow than Helicopters. | +| Sailboats are 17% fuel-efficient than Scooters. | +| Commercial airplanes are 17% noisy than Submarines. | +| Trains are 199% environmentally friendly than Speedboats. | +| Submarines are 164% fast than Electric bikes. | +| Sedans are 72% fast than Electric bikes. | +| Helicopters are 140% fast than Electric bikes. | +| Motorcycles are 146% heavy-duty than Sedans. | +| Sedans are 132% expensive than Motorcycles. | +| Sports cars are 198% slow than Private jets. | +| Motorcycles are 182% maneuverable than Sailboats. | +| Helicopters are 161% fast than Motorcycles. | +| Scooters are 45% expensive than Sedans. | +| Electric cars are 102% cost-effective than Pickup trucks. | +| Road bikes are 42% quiet than Commercial airplanes. | +| Mountain bikes are 196% versatile than Helicopters. | +| Sports cars are 139% slow than Submarines. | +| Scooters are 178% noisy than Hybrid vehicles. | +| Yachts are 187% quick to accelerate than Private jets. | +| SUVs are 38% heavy-duty than Cargo ships. | +| Yachts are 163% powerful than Hybrid vehicles. | +| Hybrid vehicles are 5% lightweight than Yachts. | +| Sailboats are 38% versatile than Hybrid vehicles. | +| Sports cars are 46% fast than Electric bikes. | +| Commercial airplanes are 25% lightweight than Cargo ships. | +| Yachts are 91% slow than Cargo ships. | +| Pickup trucks are 115% slow than Sailboats. | +| Sedans are 156% quick to accelerate than Road bikes. | +| Submarines are 176% noisy than Scooters. | +| Scooters are 90% stable than Trains. | +| Electric cars are 174% fast than Cargo ships. | +| Sedans are 64% slow than Motorcycles. | +| Mountain bikes are 57% noisy than Pickup trucks. | +| Helicopters are 158% maneuverable than Cargo ships. | +| Helicopters are 6% expensive than Scooters. | +| Helicopters are 160% lightweight than Electric bikes. | +| Cargo ships are 90% fast than Sports cars. | +| Scooters are 28% lightweight than Sailboats. | +| Electric bikes are 159% heavy-duty than Sailboats. | +| Mountain bikes are 200% efficient than Electric cars. | +| Yachts are 27% quick to accelerate than Scooters. | +| SUVs are 174% fast than Helicopters. | +| Commercial airplanes are 173% lightweight than Yachts. | +| SUVs are 114% cost-effective than Sailboats. | +| Submarines are 131% heavy-duty than Pickup trucks. | +| Yachts are 101% environmentally friendly than Pickup trucks. | +| Speedboats are 74% efficient than Helicopters. | +| Electric cars are 97% fast than Hybrid vehicles. | +| Private jets are 87% quick to accelerate than Submarines. | +| Road bikes are 1% versatile than Commercial airplanes. | +| Mountain bikes are 138% stable than Motorcycles. | +| Motorcycles are 80% environmentally friendly than Road bikes. | +| Helicopters are 162% efficient than Submarines. | +| Pickup trucks are 189% quick to accelerate than Road bikes. | +| Mountain bikes are 178% fast than Hybrid vehicles. | +| Cargo ships are 28% quick to brake than Electric bikes. | +| Pickup trucks are 124% fast than Sailboats. | +| Helicopters are 27% versatile than Sedans. | +| Speedboats are 157% quiet than Commercial airplanes. | +| Electric cars are 135% environmentally friendly than Scooters. | +| Sedans are 23% quiet than Yachts. | +| Scooters are 3% fast than Motorcycles. | +| Motorcycles are 182% lightweight than Sailboats. | +| Submarines are 89% cost-effective than Scooters. | +| Commercial airplanes are 118% versatile than Pickup trucks. | +| Helicopters are 110% quiet than Motorcycles. | +| Hybrid vehicles are 51% lightweight than Helicopters. | +| Motorcycles are 151% durable than Cargo ships. | +| Road bikes are 52% versatile than Speedboats. | +| Sailboats are 104% quick to brake than Pickup trucks. | +| Trains are 185% heavy-duty than SUVs. | +| Scooters are 64% affordable than Sedans. | +| Electric cars are 91% lightweight than Mountain bikes. | +| Scooters are 44% affordable than Speedboats. | +| Scooters are 19% stable than Cargo ships. | +| Submarines are 26% quick to brake than Hybrid vehicles. | +| Pickup trucks are 98% stable than Commercial airplanes. | +| Pickup trucks are 55% lightweight than SUVs. | +| Hybrid vehicles are 90% maneuverable than Helicopters. | +| SUVs are 138% powerful than Sports cars. | +| Trains are 187% cost-effective than Road bikes. | +| Helicopters are 134% environmentally friendly than Hybrid vehicles. | +| Yachts are 170% quiet than Motorcycles. | +| Commercial airplanes are 95% expensive than Mountain bikes. | +| Hybrid vehicles are 112% environmentally friendly than Sports cars. | +| Submarines are 78% stable than Hybrid vehicles. | +| Hybrid vehicles are 20% quick to brake than Mountain bikes. | +| Electric bikes are 33% slow than Hybrid vehicles. | +| Mountain bikes are 80% affordable than Sailboats. | +| Scooters are 105% cost-effective than Helicopters. | +| Road bikes are 159% affordable than Speedboats. | +| Electric bikes are 171% quick to brake than Private jets. | +| Sailboats are 107% expensive than Electric cars. | +| Commercial airplanes are 192% expensive than Mountain bikes. | +| Yachts are 130% fast than Scooters. | +| Mountain bikes are 167% cost-effective than Sailboats. | +| Sailboats are 198% versatile than Motorcycles. | +| Electric bikes are 9% quick to brake than Sedans. | +| Sailboats are 131% maneuverable than Motorcycles. | +| Sailboats are 139% quiet than Speedboats. | +| Scooters are 116% noisy than Commercial airplanes. | +| Sailboats are 56% versatile than Scooters. | +| Helicopters are 43% efficient than Electric bikes. | +| Scooters are 75% lightweight than Mountain bikes. | +| Sports cars are 139% environmentally friendly than Sailboats. | +| Private jets are 72% efficient than Sports cars. | +| Submarines are 174% lightweight than Cargo ships. | +| Yachts are 24% cost-effective than Scooters. | +| Submarines are 49% affordable than Sailboats. | +| Speedboats are 44% fast than Hybrid vehicles. | +| Electric cars are 67% stable than Trains. | +| Pickup trucks are 60% expensive than Cargo ships. | +| Scooters are 102% fuel-efficient than Sailboats. | +| Cargo ships are 54% slow than Hybrid vehicles. | +| Sailboats are 39% heavy-duty than Electric cars. | +| Yachts are 141% expensive than Speedboats. | +| Submarines are 181% affordable than Sports cars. | +| Cargo ships are 119% stable than Yachts. | +| Scooters are 90% durable than Electric cars. | +| Pickup trucks are 79% heavy-duty than Sedans. | +| Private jets are 82% expensive than SUVs. | +| Trains are 168% affordable than SUVs. | +| Hybrid vehicles are 41% lightweight than Scooters. | +| Mountain bikes are 91% heavy-duty than Hybrid vehicles. | +| Electric cars are 75% quick to accelerate than Private jets. | +| Helicopters are 110% efficient than Speedboats. | +| Motorcycles are 123% powerful than Commercial airplanes. | +| Sedans are 32% heavy-duty than Sailboats. | +| Pickup trucks are 29% stable than Speedboats. | +| Road bikes are 150% stable than Motorcycles. | +| Yachts are 139% fuel-efficient than Pickup trucks. | +| Private jets are 63% slow than Electric cars. | +| Sedans are 173% noisy than Trains. | +| Motorcycles are 23% quiet than Cargo ships. | +| SUVs are 5% powerful than Sailboats. | +| Sailboats are 133% fuel-efficient than Trains. | +| Hybrid vehicles are 89% expensive than Trains. | +| Helicopters are 175% expensive than Mountain bikes. | +| Hybrid vehicles are 94% lightweight than SUVs. | +| Helicopters are 71% quiet than SUVs. | +| Sailboats are 90% powerful than SUVs. | +| Trains are 149% lightweight than Scooters. | +| Mountain bikes are 150% stable than Road bikes. | +| Speedboats are 198% durable than Scooters. | +| Electric bikes are 43% powerful than Speedboats. | +| Road bikes are 193% quick to accelerate than Sailboats. | +| Pickup trucks are 39% maneuverable than Electric bikes. | +| Sports cars are 119% heavy-duty than Sedans. | +| Yachts are 170% durable than Helicopters. | +| Private jets are 46% expensive than Scooters. | +| Electric bikes are 44% fast than Submarines. | +| Pickup trucks are 191% heavy-duty than Sports cars. | +| Electric bikes are 120% quick to accelerate than Hybrid vehicles. | +| Road bikes are 44% quick to brake than Cargo ships. | +| Submarines are 111% durable than Yachts. | +| Scooters are 7% quick to brake than Commercial airplanes. | +| Hybrid vehicles are 63% lightweight than Electric bikes. | +| Road bikes are 94% cost-effective than Cargo ships. | +| Sailboats are 132% slow than Motorcycles. | +| Sedans are 62% environmentally friendly than Commercial airplanes. | +| Trains are 51% maneuverable than Speedboats. | +| Motorcycles are 92% noisy than Cargo ships. | +| Sports cars are 25% lightweight than Sedans. | +| Speedboats are 30% affordable than Pickup trucks. | +| Submarines are 46% cost-effective than Yachts. | +| Commercial airplanes are 67% durable than Sailboats. | +| Hybrid vehicles are 175% slow than Yachts. | +| Sports cars are 23% noisy than Scooters. | +| Private jets are 46% environmentally friendly than Cargo ships. | +| Electric bikes are 155% heavy-duty than Yachts. | +| Trains are 40% lightweight than Yachts. | +| Electric cars are 96% stable than Trains. | +| Private jets are 103% fuel-efficient than Speedboats. | +| Road bikes are 176% fuel-efficient than SUVs. | +| SUVs are 105% quick to brake than Pickup trucks. | +| Trains are 75% versatile than Private jets. | +| Electric cars are 87% heavy-duty than Speedboats. | +| Helicopters are 4% environmentally friendly than Sailboats. | +| Commercial airplanes are 33% stable than Sailboats. | +| Submarines are 19% quick to brake than Electric bikes. | +| Electric cars are 112% fast than Private jets. | +| Cargo ships are 54% stable than Sailboats. | +| Road bikes are 153% fuel-efficient than Yachts. | +| Cargo ships are 106% stable than Sailboats. | +| Scooters are 162% affordable than Speedboats. | +| Electric bikes are 56% maneuverable than SUVs. | +| Yachts are 176% slow than Sports cars. | +| Sailboats are 19% versatile than Road bikes. | +| Motorcycles are 31% fuel-efficient than Submarines. | +| Yachts are 56% fuel-efficient than Trains. | +| Helicopters are 150% stable than Motorcycles. | +| Sedans are 45% noisy than Hybrid vehicles. | +| Road bikes are 159% cost-effective than Yachts. | +| Cargo ships are 187% maneuverable than Motorcycles. | +| Scooters are 184% quiet than Sedans. | +| Sports cars are 51% quick to accelerate than Scooters. | +| Private jets are 81% durable than Helicopters. | +| Hybrid vehicles are 143% heavy-duty than Electric cars. | +| Sailboats are 85% maneuverable than Yachts. | +| Pickup trucks are 124% maneuverable than Scooters. | +| SUVs are 2% cost-effective than Yachts. | +| Private jets are 153% quick to accelerate than Motorcycles. | +| Private jets are 199% affordable than Yachts. | +| Cargo ships are 151% efficient than Electric bikes. | +| SUVs are 11% quick to brake than Mountain bikes. | +| Yachts are 30% lightweight than Helicopters. | +| Electric cars are 4% expensive than Trains. | +| Sports cars are 141% quick to brake than Pickup trucks. | +| Private jets are 146% expensive than Sailboats. | +| Hybrid vehicles are 89% noisy than Trains. | +| Sports cars are 79% quick to accelerate than Helicopters. | +| Electric bikes are 86% expensive than SUVs. | +| Speedboats are 31% slow than Pickup trucks. | +| Yachts are 42% cost-effective than Commercial airplanes. | +| Pickup trucks are 34% versatile than Electric cars. | +| Electric bikes are 9% quick to brake than Trains. | +| Submarines are 130% slow than Motorcycles. | +| Scooters are 73% cost-effective than Mountain bikes. | +| Sedans are 55% heavy-duty than Private jets. | +| Motorcycles are 3% efficient than Sports cars. | +| Electric cars are 73% environmentally friendly than Cargo ships. | +| Sports cars are 58% expensive than Private jets. | +| Yachts are 57% heavy-duty than Road bikes. | +| Electric cars are 150% fast than Scooters. | +| Electric bikes are 136% heavy-duty than Hybrid vehicles. | +| Mountain bikes are 12% slow than Electric bikes. | +| Cargo ships are 122% slow than Scooters. | +| Sedans are 170% quiet than Hybrid vehicles. | +| Speedboats are 170% quick to accelerate than Sailboats. | +| Hybrid vehicles are 13% quick to brake than Private jets. | +| Hybrid vehicles are 96% quick to brake than Sports cars. | +| Electric bikes are 129% powerful than Scooters. | +| Pickup trucks are 129% lightweight than SUVs. | +| Electric cars are 113% quick to brake than Sailboats. | +| Scooters are 78% expensive than Trains. | +| Hybrid vehicles are 140% versatile than Helicopters. | +| Yachts are 138% quiet than Pickup trucks. | +| Electric bikes are 156% cost-effective than Private jets. | +| Road bikes are 150% environmentally friendly than Private jets. | +| Speedboats are 61% powerful than Trains. | +| Speedboats are 88% slow than Road bikes. | +| Scooters are 175% quiet than Electric cars. | +| Private jets are 108% noisy than Speedboats. | +| Road bikes are 19% maneuverable than Helicopters. | +| Speedboats are 167% quiet than Road bikes. | +| Cargo ships are 184% quick to accelerate than Sailboats. | +| Motorcycles are 90% quick to brake than SUVs. | +| Yachts are 168% quick to accelerate than Trains. | +| SUVs are 74% slow than Helicopters. | +| Sailboats are 154% slow than Electric cars. | +| Cargo ships are 173% noisy than Motorcycles. | +| Private jets are 98% affordable than Sports cars. | +| Pickup trucks are 13% versatile than Scooters. | +| Speedboats are 167% maneuverable than Commercial airplanes. | +| Submarines are 5% expensive than Electric bikes. | +| Pickup trucks are 80% affordable than Submarines. | +| Hybrid vehicles are 150% quick to brake than Sports cars. | +| Helicopters are 42% heavy-duty than Road bikes. | +| Sailboats are 87% environmentally friendly than Helicopters. | +| Road bikes are 60% expensive than Electric cars. | +| Speedboats are 9% versatile than Scooters. | +| Yachts are 7% cost-effective than Pickup trucks. | +| Speedboats are 85% environmentally friendly than Trains. | +| Pickup trucks are 184% cost-effective than Private jets. | +| Trains are 113% heavy-duty than Sedans. | +| Speedboats are 35% quiet than Electric cars. | +| Private jets are 198% cost-effective than Mountain bikes. | +| Private jets are 126% expensive than Sailboats. | +| SUVs are 143% cost-effective than Speedboats. | +| Private jets are 155% stable than Trains. | +| Helicopters are 123% expensive than Private jets. | +| Speedboats are 70% environmentally friendly than Private jets. | +| Sailboats are 124% durable than Scooters. | +| Yachts are 101% quick to accelerate than Cargo ships. | +| Pickup trucks are 149% quick to brake than Hybrid vehicles. | +| Submarines are 114% quiet than Electric cars. | +| Sedans are 79% expensive than Road bikes. | +| Helicopters are 187% cost-effective than Hybrid vehicles. | +| Road bikes are 138% cost-effective than Cargo ships. | +| Yachts are 38% noisy than Private jets. | +| Speedboats are 66% lightweight than Submarines. | +| Helicopters are 119% heavy-duty than Mountain bikes. | +| Sedans are 12% stable than Speedboats. | +| Helicopters are 118% noisy than Pickup trucks. | +| Sports cars are 34% durable than SUVs. | +| Cargo ships are 46% slow than Sailboats. | +| Speedboats are 82% powerful than Private jets. | +| Electric cars are 69% cost-effective than Hybrid vehicles. | +| Submarines are 130% powerful than Electric cars. | +| Road bikes are 143% slow than Helicopters. | +| Commercial airplanes are 140% maneuverable than Sports cars. | +| Yachts are 50% noisy than Private jets. | +| Sports cars are 102% heavy-duty than Motorcycles. | +| Speedboats are 98% slow than Cargo ships. | +| Trains are 38% efficient than Scooters. | +| Sailboats are 69% noisy than Electric cars. | +| Sedans are 137% lightweight than Commercial airplanes. | +| Mountain bikes are 144% environmentally friendly than Road bikes. | +| Motorcycles are 71% quick to brake than Speedboats. | +| Motorcycles are 21% quiet than Sailboats. | +| Helicopters are 38% maneuverable than Sailboats. | +| Electric bikes are 163% noisy than Sedans. | +| Electric cars are 166% fuel-efficient than Yachts. | +| Cargo ships are 147% fuel-efficient than Electric bikes. | +| Private jets are 113% heavy-duty than Pickup trucks. | +| Electric cars are 29% quiet than Scooters. | +| Helicopters are 20% quick to accelerate than Mountain bikes. | +| Speedboats are 114% fuel-efficient than Trains. | +| Submarines are 102% heavy-duty than Electric bikes. | +| Motorcycles are 50% cost-effective than Sailboats. | +| Electric cars are 71% heavy-duty than Commercial airplanes. | +| Commercial airplanes are 159% cost-effective than Pickup trucks. | +| Trains are 64% stable than Scooters. | +| Sports cars are 37% quick to accelerate than Commercial airplanes. | +| Trains are 183% fast than Road bikes. | +| SUVs are 144% slow than Sedans. | +| Pickup trucks are 134% stable than Hybrid vehicles. | +| Electric cars are 82% affordable than Scooters. | +| Submarines are 103% efficient than SUVs. | +| Motorcycles are 149% maneuverable than Sailboats. | +| Mountain bikes are 165% quick to accelerate than Pickup trucks. | +| Road bikes are 44% stable than Electric bikes. | +| Submarines are 15% expensive than Hybrid vehicles. | +| Electric bikes are 9% fuel-efficient than Pickup trucks. | +| Speedboats are 182% quick to accelerate than Electric cars. | +| Scooters are 37% affordable than Hybrid vehicles. | +| Electric cars are 139% quick to brake than Yachts. | +| SUVs are 155% lightweight than Electric bikes. | +| Sedans are 8% heavy-duty than Helicopters. | +| Scooters are 24% expensive than Trains. | +| Commercial airplanes are 118% noisy than SUVs. | +| SUVs are 174% lightweight than Sedans. | +| Trains are 70% affordable than Motorcycles. | +| Yachts are 69% lightweight than Commercial airplanes. | +| Sailboats are 98% powerful than Hybrid vehicles. | +| SUVs are 148% stable than Sailboats. | +| Submarines are 58% durable than Motorcycles. | +| Sports cars are 140% stable than Private jets. | +| Submarines are 30% lightweight than Road bikes. | +| Hybrid vehicles are 67% environmentally friendly than Road bikes. | +| Electric cars are 23% efficient than Speedboats. | +| Electric bikes are 75% slow than Motorcycles. | +| Trains are 140% lightweight than Helicopters. | +| Sports cars are 150% quiet than Scooters. | +| Speedboats are 166% environmentally friendly than Sports cars. | +| Road bikes are 171% heavy-duty than Private jets. | +| Electric bikes are 27% versatile than Private jets. | +| Sports cars are 2% cost-effective than Electric bikes. | +| Road bikes are 44% quick to brake than Trains. | +| Sedans are 47% maneuverable than Trains. | +| Mountain bikes are 87% cost-effective than Yachts. | +| Scooters are 176% quick to brake than Trains. | +| Sedans are 120% cost-effective than Speedboats. | +| Motorcycles are 199% cost-effective than Hybrid vehicles. | +| Cargo ships are 119% efficient than Trains. | +| Mountain bikes are 98% cost-effective than Helicopters. | +| Scooters are 35% noisy than Electric bikes. | +| Yachts are 132% versatile than Road bikes. | +| Speedboats are 113% quick to brake than Sports cars. | +| Yachts are 150% fast than Sports cars. | +| SUVs are 94% quiet than Commercial airplanes. | +| Road bikes are 61% quick to brake than Helicopters. | +| Private jets are 85% fast than Commercial airplanes. | +| Commercial airplanes are 127% maneuverable than Motorcycles. | +| SUVs are 167% stable than Sedans. | +| Helicopters are 126% fast than Speedboats. | +| Electric cars are 89% noisy than Scooters. | +| Private jets are 78% versatile than Scooters. | +| Electric bikes are 120% affordable than Hybrid vehicles. | +| Submarines are 72% fast than SUVs. | +| Trains are 20% versatile than Electric cars. | +| Trains are 129% slow than Road bikes. | +| Submarines are 90% versatile than Electric cars. | +| Cargo ships are 39% quick to brake than Hybrid vehicles. | +| Sailboats are 80% lightweight than Electric bikes. | +| Submarines are 199% environmentally friendly than Pickup trucks. | +| Private jets are 114% cost-effective than Yachts. | +| Trains are 21% slow than Speedboats. | +| SUVs are 52% slow than Commercial airplanes. | +| Scooters are 28% affordable than Private jets. | +| Private jets are 90% fuel-efficient than Cargo ships. | +| Road bikes are 89% efficient than Pickup trucks. | +| Pickup trucks are 91% lightweight than Cargo ships. | +| Pickup trucks are 134% versatile than Hybrid vehicles. | +| Hybrid vehicles are 108% durable than Submarines. | +| Private jets are 78% durable than Helicopters. | +| Sedans are 198% expensive than SUVs. | +| Electric cars are 182% quick to accelerate than Electric bikes. | +| Motorcycles are 66% efficient than Scooters. | +| Submarines are 177% expensive than Motorcycles. | +| Hybrid vehicles are 104% fuel-efficient than Submarines. | +| SUVs are 163% noisy than Yachts. | +| Road bikes are 191% durable than Sailboats. | +| Private jets are 74% heavy-duty than Scooters. | +| Commercial airplanes are 51% noisy than SUVs. | +| Mountain bikes are 89% affordable than Hybrid vehicles. | +| Submarines are 167% lightweight than Mountain bikes. | +| Sports cars are 190% cost-effective than Motorcycles. | +| Scooters are 15% stable than Electric cars. | +| SUVs are 161% fast than Helicopters. | +| Submarines are 74% fast than Private jets. | +| Yachts are 97% lightweight than Submarines. | +| Electric cars are 176% affordable than Sedans. | +| Road bikes are 79% versatile than Sports cars. | +| Speedboats are 112% versatile than Electric cars. | +| Sports cars are 148% cost-effective than Submarines. | +| Sailboats are 110% quick to brake than Yachts. | +| SUVs are 173% efficient than Sedans. | +| Electric bikes are 131% fast than Commercial airplanes. | +| Speedboats are 25% fast than Scooters. | +| Pickup trucks are 13% fast than Scooters. | +| Yachts are 20% slow than Speedboats. | +| Sailboats are 159% cost-effective than Cargo ships. | +| Cargo ships are 120% environmentally friendly than Private jets. | +| Pickup trucks are 133% versatile than Trains. | +| Helicopters are 10% quiet than Commercial airplanes. | +| SUVs are 15% durable than Private jets. | +| Submarines are 35% quiet than SUVs. | +| Trains are 132% maneuverable than Yachts. | +| Helicopters are 143% stable than Hybrid vehicles. | +| Scooters are 169% heavy-duty than Commercial airplanes. | +| Yachts are 96% expensive than Speedboats. | +| Helicopters are 126% quick to accelerate than Scooters. | +| Private jets are 143% noisy than SUVs. | +| Motorcycles are 141% noisy than Yachts. | +| Scooters are 45% quick to accelerate than Pickup trucks. | +| Scooters are 155% lightweight than Private jets. | +| Electric cars are 66% fast than Yachts. | +| Sports cars are 88% cost-effective than Cargo ships. | +| Pickup trucks are 93% heavy-duty than Commercial airplanes. | +| Commercial airplanes are 99% lightweight than Private jets. | +| Sports cars are 138% heavy-duty than Electric bikes. | +| SUVs are 44% slow than Commercial airplanes. | +| Cargo ships are 81% efficient than Electric cars. | +| Yachts are 104% cost-effective than Electric cars. | +| Submarines are 65% powerful than Scooters. | +| Yachts are 69% noisy than Road bikes. | +| Road bikes are 185% environmentally friendly than Speedboats. | +| Helicopters are 200% stable than SUVs. | +| Road bikes are 34% cost-effective than Electric cars. | +| Helicopters are 192% affordable than Road bikes. | +| SUVs are 92% versatile than Motorcycles. | +| Sedans are 16% affordable than Cargo ships. | +| Road bikes are 118% fuel-efficient than Submarines. | +| Commercial airplanes are 176% environmentally friendly than Helicopters. | +| Trains are 37% lightweight than Pickup trucks. | +| Motorcycles are 21% efficient than Commercial airplanes. | +| Speedboats are 29% heavy-duty than Commercial airplanes. | +| Pickup trucks are 118% quiet than Sailboats. | +| Mountain bikes are 72% stable than Speedboats. | +| Mountain bikes are 109% quiet than Private jets. | +| Electric bikes are 181% stable than Sailboats. | +| Scooters are 13% lightweight than Electric bikes. | +| Pickup trucks are 84% fuel-efficient than Scooters. | +| Pickup trucks are 164% environmentally friendly than Road bikes. | +| Helicopters are 7% slow than Cargo ships. | +| Submarines are 50% quiet than Sedans. | +| Sports cars are 157% environmentally friendly than Electric bikes. | +| Trains are 17% slow than Hybrid vehicles. | +| SUVs are 76% powerful than Sedans. | +| Electric bikes are 157% quick to accelerate than Sailboats. | +| Speedboats are 114% powerful than Sailboats. | +| Pickup trucks are 63% stable than Electric cars. | +| Scooters are 119% versatile than Speedboats. | +| Electric cars are 90% fast than Trains. | +| Scooters are 163% expensive than Mountain bikes. | +| Yachts are 132% affordable than Commercial airplanes. | +| Hybrid vehicles are 68% quiet than Sailboats. | +| Hybrid vehicles are 113% powerful than Motorcycles. | +| Mountain bikes are 194% heavy-duty than Sailboats. | +| Scooters are 82% versatile than Sailboats. | +| Sports cars are 149% maneuverable than Motorcycles. | +| Scooters are 193% quiet than Helicopters. | +| Sports cars are 195% quick to brake than Scooters. | +| Hybrid vehicles are 82% lightweight than Yachts. | +| SUVs are 131% affordable than Scooters. | +| Yachts are 108% stable than Motorcycles. | +| Road bikes are 102% lightweight than Motorcycles. | +| Speedboats are 50% efficient than Mountain bikes. | +| Cargo ships are 200% maneuverable than Sports cars. | +| Electric cars are 17% lightweight than Cargo ships. | +| Speedboats are 163% cost-effective than Sailboats. | +| Hybrid vehicles are 4% environmentally friendly than Sailboats. | +| Pickup trucks are 198% lightweight than Mountain bikes. | +| Submarines are 151% lightweight than Electric bikes. | +| Cargo ships are 144% quick to brake than SUVs. | +| Yachts are 189% maneuverable than Commercial airplanes. | +| Motorcycles are 87% expensive than Electric cars. | +| Trains are 25% powerful than Submarines. | +| SUVs are 75% slow than Sports cars. | +| Sailboats are 126% stable than Submarines. | +| Mountain bikes are 28% versatile than Road bikes. | +| Sedans are 11% stable than Scooters. | +| Mountain bikes are 29% cost-effective than Pickup trucks. | +| Scooters are 167% quiet than Sports cars. | +| Commercial airplanes are 21% quick to accelerate than Cargo ships. | +| Commercial airplanes are 168% powerful than Private jets. | +| Motorcycles are 200% cost-effective than Cargo ships. | +| Submarines are 183% quiet than Sports cars. | +| Mountain bikes are 80% quiet than Private jets. | +| Sports cars are 163% noisy than Road bikes. | +| Speedboats are 17% fuel-efficient than Motorcycles. | +| Yachts are 116% stable than Submarines. | +| Trains are 109% expensive than Scooters. | +| Pickup trucks are 155% quick to accelerate than Hybrid vehicles. | +| Sports cars are 59% fast than Commercial airplanes. | +| Commercial airplanes are 23% environmentally friendly than Road bikes. | +| SUVs are 149% slow than Electric bikes. | +| SUVs are 176% durable than Hybrid vehicles. | +| Sports cars are 48% environmentally friendly than Commercial airplanes. | +| Private jets are 112% heavy-duty than Speedboats. | +| Helicopters are 66% quick to accelerate than Pickup trucks. | +| Submarines are 87% affordable than Mountain bikes. | +| SUVs are 101% stable than Motorcycles. | +| Scooters are 161% fuel-efficient than Submarines. | +| Sailboats are 141% efficient than Sports cars. | +| Submarines are 35% quiet than Speedboats. | +| Submarines are 36% quiet than Pickup trucks. | +| Private jets are 62% quick to accelerate than Commercial airplanes. | +| Hybrid vehicles are 128% fuel-efficient than Pickup trucks. | +| SUVs are 19% durable than Yachts. | +| Commercial airplanes are 32% affordable than Submarines. | +| Sports cars are 62% maneuverable than Road bikes. | +| Sedans are 164% stable than Speedboats. | +| Scooters are 166% fast than SUVs. | +| Electric bikes are 160% quick to accelerate than Scooters. | +| Electric bikes are 136% stable than Sedans. | +| SUVs are 50% stable than Speedboats. | +| Sailboats are 102% stable than Hybrid vehicles. | +| Mountain bikes are 138% cost-effective than Road bikes. | +| Speedboats are 133% affordable than Commercial airplanes. | +| Sailboats are 80% efficient than Helicopters. | +| Electric bikes are 194% maneuverable than Electric cars. | +| Electric cars are 176% stable than Motorcycles. | +| Speedboats are 16% efficient than Trains. | +| Helicopters are 36% quick to brake than Sports cars. | +| Speedboats are 139% quiet than Sailboats. | +| Commercial airplanes are 196% heavy-duty than SUVs. | +| Speedboats are 98% noisy than Motorcycles. | +| Electric bikes are 96% quick to accelerate than Road bikes. | +| Hybrid vehicles are 36% durable than Electric cars. | +| SUVs are 147% stable than Sedans. | +| Road bikes are 102% affordable than Helicopters. | +| Yachts are 17% slow than SUVs. | +| Pickup trucks are 158% noisy than Electric cars. | +| Scooters are 162% fuel-efficient than Yachts. | +| Sports cars are 102% slow than SUVs. | +| Mountain bikes are 75% slow than Hybrid vehicles. | +| Electric cars are 177% versatile than Sports cars. | +| SUVs are 120% maneuverable than Electric cars. | +| Motorcycles are 128% powerful than Sports cars. | +| Commercial airplanes are 20% quiet than Cargo ships. | +| SUVs are 135% quick to accelerate than Commercial airplanes. | +| Electric bikes are 32% noisy than Commercial airplanes. | +| SUVs are 101% expensive than Hybrid vehicles. | +| Electric bikes are 147% quick to accelerate than Submarines. | +| Road bikes are 92% lightweight than Yachts. | +| Cargo ships are 17% lightweight than Motorcycles. | +| Cargo ships are 62% durable than Sailboats. | +| Electric bikes are 186% slow than Yachts. | +| Sports cars are 81% expensive than Pickup trucks. | +| Electric cars are 26% powerful than Helicopters. | +| Helicopters are 70% quick to brake than Mountain bikes. | +| Sedans are 37% heavy-duty than Road bikes. | +| Road bikes are 105% noisy than Electric cars. | +| Commercial airplanes are 105% lightweight than Helicopters. | +| Electric bikes are 52% quick to brake than Commercial airplanes. | +| Pickup trucks are 180% affordable than Electric cars. | +| Road bikes are 113% expensive than Private jets. | +| Submarines are 69% quick to brake than Sedans. | +| SUVs are 95% efficient than Hybrid vehicles. | +| Trains are 189% quick to accelerate than Pickup trucks. | +| SUVs are 186% noisy than Motorcycles. | +| Motorcycles are 24% quick to brake than Scooters. | +| Scooters are 44% environmentally friendly than Mountain bikes. | +| Motorcycles are 105% affordable than Electric bikes. | +| Speedboats are 110% fast than Cargo ships. | +| Trains are 170% lightweight than SUVs. | +| Pickup trucks are 194% expensive than Private jets. | +| Sailboats are 152% powerful than SUVs. | +| Speedboats are 147% lightweight than Sports cars. | +| Submarines are 25% versatile than Speedboats. | +| Scooters are 155% maneuverable than Pickup trucks. | +| Helicopters are 125% noisy than Cargo ships. | +| Electric cars are 55% heavy-duty than Sailboats. | +| Private jets are 177% maneuverable than Sailboats. | +| Electric cars are 129% cost-effective than SUVs. | +| Speedboats are 143% cost-effective than Motorcycles. | +| Motorcycles are 120% noisy than Cargo ships. | +| Trains are 140% affordable than Road bikes. | +| SUVs are 126% cost-effective than Speedboats. | +| Trains are 85% quick to brake than Speedboats. | +| Sedans are 1% heavy-duty than Hybrid vehicles. | +| Helicopters are 193% quiet than Submarines. | +| Submarines are 112% fast than Pickup trucks. | +| Scooters are 166% durable than Sailboats. | +| Private jets are 43% quick to accelerate than SUVs. | +| Electric cars are 63% powerful than Scooters. | +| Yachts are 169% expensive than Helicopters. | +| Motorcycles are 138% expensive than Sedans. | +| Cargo ships are 198% durable than Sailboats. | +| Submarines are 73% stable than Helicopters. | +| Scooters are 137% stable than Cargo ships. | +| Hybrid vehicles are 105% fuel-efficient than Electric bikes. | +| Mountain bikes are 85% powerful than Trains. | +| Sailboats are 167% environmentally friendly than Electric bikes. | +| Speedboats are 152% lightweight than Yachts. | +| Commercial airplanes are 119% quick to brake than Helicopters. | +| Electric cars are 69% quick to accelerate than Mountain bikes. | +| Motorcycles are 128% cost-effective than Hybrid vehicles. | +| Mountain bikes are 84% lightweight than SUVs. | +| Scooters are 16% noisy than Sedans. | +| Helicopters are 7% quick to accelerate than Electric cars. | +| Speedboats are 75% maneuverable than Sailboats. | +| Commercial airplanes are 88% cost-effective than Pickup trucks. | +| Mountain bikes are 66% environmentally friendly than Electric bikes. | \ No newline at end of file diff --git a/pkg/vectordb/pinecone.go b/pkg/vectordb/pinecone.go index 663dd0b..f4de203 100644 --- a/pkg/vectordb/pinecone.go +++ b/pkg/vectordb/pinecone.go @@ -25,7 +25,7 @@ func NewPineconeVectorDB(threshold float32, topKResultsNumber int, index string, } } -func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings []*domain.Embeddings) (int, error) { +func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings []*domain.Embeddings, extraMetadata map[string]interface{}) (int, error) { idx, err := db.client.DescribeIndex(ctx, db.index) if err != nil { return 0, err @@ -47,6 +47,15 @@ func (db *PineconeVectorDB) StoreEmbeddings(ctx context.Context, embeddings []*d return 0, err } + if len(extraMetadata) > 0 { + for key, value := range extraMetadata { + md.Fields[key], err = structpb.NewValue(value) + if err != nil { + return 0, err + } + } + } + vectorToFloat32 := helpers.Float64ToFloat32(embedding.Embeddings) vectors[i] = &pinecone.Vector{ From 5bedbb5c8acaad44c97f783c8c979f9c773a5143 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 18 Jun 2025 13:09:47 +0300 Subject: [PATCH 45/52] Move http stuff to dedicated folders --- cmd/http/main.go | 10 +++++----- .../chatSessions/createChatSessionHandler.go | 0 .../createChatSessionHandler_test.go | 0 .../handlers/{ => http}/chatSessions/dto.go | 0 .../chatSessions/getChatSessionHandler.go | 0 .../getChatSessionHandler_test.go | 0 .../chatSessions/sendMessageHandler.go | 0 .../chatSessions/sendMessageHandler_test.go | 0 .../chatSessions/submitFeedbackHandler.go | 0 .../submitFeedbackHandler_test.go | 0 .../handlers/{ => http}/healthCheckHandler.go | 2 +- .../handlers/{ => http}/jwtClaimsHandler.go | 2 +- internal/handlers/{ => http}/middleware.go | 2 +- pkg/server/{ => http}/routes.go | 20 +++++++++---------- pkg/server/{ => http}/server.go | 2 +- 15 files changed, 19 insertions(+), 19 deletions(-) rename internal/handlers/{ => http}/chatSessions/createChatSessionHandler.go (100%) rename internal/handlers/{ => http}/chatSessions/createChatSessionHandler_test.go (100%) rename internal/handlers/{ => http}/chatSessions/dto.go (100%) rename internal/handlers/{ => http}/chatSessions/getChatSessionHandler.go (100%) rename internal/handlers/{ => http}/chatSessions/getChatSessionHandler_test.go (100%) rename internal/handlers/{ => http}/chatSessions/sendMessageHandler.go (100%) rename internal/handlers/{ => http}/chatSessions/sendMessageHandler_test.go (100%) rename internal/handlers/{ => http}/chatSessions/submitFeedbackHandler.go (100%) rename internal/handlers/{ => http}/chatSessions/submitFeedbackHandler_test.go (100%) rename internal/handlers/{ => http}/healthCheckHandler.go (98%) rename internal/handlers/{ => http}/jwtClaimsHandler.go (99%) rename internal/handlers/{ => http}/middleware.go (98%) rename pkg/server/{ => http}/routes.go (74%) rename pkg/server/{ => http}/server.go (99%) diff --git a/cmd/http/main.go b/cmd/http/main.go index cdb3752..11887e2 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -10,7 +10,7 @@ import ( "github.com/loukaspe/jedi-team-challenge/pkg/chunks" "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" "github.com/loukaspe/jedi-team-challenge/pkg/logger" - "github.com/loukaspe/jedi-team-challenge/pkg/server" + http2 "github.com/loukaspe/jedi-team-challenge/pkg/server/http" "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" "github.com/openai/openai-go" "github.com/openai/openai-go/option" @@ -28,13 +28,13 @@ func main() { ctx := context.Background() getEnv() - encoder := getEncoder() + //encoder := getEncoder() client := getOpenAIClient() - chunker := getChunker(encoder) + //chunker := getChunker(encoder) embedder := getEmbedder(&client) pineconeVectorDB := getPineconeVectorDB() - inputPeopleKnowledgeBase(ctx, chunker, embedder, pineconeVectorDB) + //inputPeopleKnowledgeBase(ctx, chunker, embedder, pineconeVectorDB) logger := logger.NewLogger(ctx) router := mux.NewRouter() @@ -44,7 +44,7 @@ func main() { } db := getDB() - server := server.NewServer(db, router, httpServer, logger, &client, embedder, pineconeVectorDB) + server := http2.NewServer(db, router, httpServer, logger, &client, embedder, pineconeVectorDB) server.Run() } diff --git a/internal/handlers/chatSessions/createChatSessionHandler.go b/internal/handlers/http/chatSessions/createChatSessionHandler.go similarity index 100% rename from internal/handlers/chatSessions/createChatSessionHandler.go rename to internal/handlers/http/chatSessions/createChatSessionHandler.go diff --git a/internal/handlers/chatSessions/createChatSessionHandler_test.go b/internal/handlers/http/chatSessions/createChatSessionHandler_test.go similarity index 100% rename from internal/handlers/chatSessions/createChatSessionHandler_test.go rename to internal/handlers/http/chatSessions/createChatSessionHandler_test.go diff --git a/internal/handlers/chatSessions/dto.go b/internal/handlers/http/chatSessions/dto.go similarity index 100% rename from internal/handlers/chatSessions/dto.go rename to internal/handlers/http/chatSessions/dto.go diff --git a/internal/handlers/chatSessions/getChatSessionHandler.go b/internal/handlers/http/chatSessions/getChatSessionHandler.go similarity index 100% rename from internal/handlers/chatSessions/getChatSessionHandler.go rename to internal/handlers/http/chatSessions/getChatSessionHandler.go diff --git a/internal/handlers/chatSessions/getChatSessionHandler_test.go b/internal/handlers/http/chatSessions/getChatSessionHandler_test.go similarity index 100% rename from internal/handlers/chatSessions/getChatSessionHandler_test.go rename to internal/handlers/http/chatSessions/getChatSessionHandler_test.go diff --git a/internal/handlers/chatSessions/sendMessageHandler.go b/internal/handlers/http/chatSessions/sendMessageHandler.go similarity index 100% rename from internal/handlers/chatSessions/sendMessageHandler.go rename to internal/handlers/http/chatSessions/sendMessageHandler.go diff --git a/internal/handlers/chatSessions/sendMessageHandler_test.go b/internal/handlers/http/chatSessions/sendMessageHandler_test.go similarity index 100% rename from internal/handlers/chatSessions/sendMessageHandler_test.go rename to internal/handlers/http/chatSessions/sendMessageHandler_test.go diff --git a/internal/handlers/chatSessions/submitFeedbackHandler.go b/internal/handlers/http/chatSessions/submitFeedbackHandler.go similarity index 100% rename from internal/handlers/chatSessions/submitFeedbackHandler.go rename to internal/handlers/http/chatSessions/submitFeedbackHandler.go diff --git a/internal/handlers/chatSessions/submitFeedbackHandler_test.go b/internal/handlers/http/chatSessions/submitFeedbackHandler_test.go similarity index 100% rename from internal/handlers/chatSessions/submitFeedbackHandler_test.go rename to internal/handlers/http/chatSessions/submitFeedbackHandler_test.go diff --git a/internal/handlers/healthCheckHandler.go b/internal/handlers/http/healthCheckHandler.go similarity index 98% rename from internal/handlers/healthCheckHandler.go rename to internal/handlers/http/healthCheckHandler.go index 9495be0..2ee4b79 100644 --- a/internal/handlers/healthCheckHandler.go +++ b/internal/handlers/http/healthCheckHandler.go @@ -1,4 +1,4 @@ -package handlers +package http import ( "encoding/json" diff --git a/internal/handlers/jwtClaimsHandler.go b/internal/handlers/http/jwtClaimsHandler.go similarity index 99% rename from internal/handlers/jwtClaimsHandler.go rename to internal/handlers/http/jwtClaimsHandler.go index dc0746a..8395c18 100644 --- a/internal/handlers/jwtClaimsHandler.go +++ b/internal/handlers/http/jwtClaimsHandler.go @@ -1,4 +1,4 @@ -package handlers +package http import ( "encoding/json" diff --git a/internal/handlers/middleware.go b/internal/handlers/http/middleware.go similarity index 98% rename from internal/handlers/middleware.go rename to internal/handlers/http/middleware.go index 80f7d1a..b50c128 100644 --- a/internal/handlers/middleware.go +++ b/internal/handlers/http/middleware.go @@ -1,4 +1,4 @@ -package handlers +package http import ( "github.com/loukaspe/jedi-team-challenge/internal/core/domain" diff --git a/pkg/server/routes.go b/pkg/server/http/routes.go similarity index 74% rename from pkg/server/routes.go rename to pkg/server/http/routes.go index 0d22ebe..0aa02d6 100644 --- a/pkg/server/routes.go +++ b/pkg/server/http/routes.go @@ -1,9 +1,9 @@ -package server +package http import ( "github.com/loukaspe/jedi-team-challenge/internal/core/services" - "github.com/loukaspe/jedi-team-challenge/internal/handlers" - "github.com/loukaspe/jedi-team-challenge/internal/handlers/chatSessions" + http2 "github.com/loukaspe/jedi-team-challenge/internal/handlers/http" + chatSessions2 "github.com/loukaspe/jedi-team-challenge/internal/handlers/http/chatSessions" "github.com/loukaspe/jedi-team-challenge/internal/repositories" "github.com/loukaspe/jedi-team-challenge/pkg/auth" @@ -30,7 +30,7 @@ import ( // @produce json func (s *Server) initializeRoutes() { // health check - healthCheckHandler := handlers.NewHealthCheckHandler(s.DB) + healthCheckHandler := http2.NewHealthCheckHandler(s.DB) s.router.HandleFunc("/health-check", healthCheckHandler.HealthCheckController).Methods("GET") // auth @@ -39,8 +39,8 @@ func (s *Server) initializeRoutes() { os.Getenv("JWT_SIGNING_METHOD"), ) jwtService := services.NewJwtService(jwtMechanism) - jwtMiddleware := handlers.NewAuthenticationMw(jwtMechanism) - jwtHandler := handlers.NewJwtClaimsHandler(jwtService, s.logger) + jwtMiddleware := http2.NewAuthenticationMw(jwtMechanism) + jwtHandler := http2.NewJwtClaimsHandler(jwtService, s.logger) s.router.HandleFunc("/token", jwtHandler.JwtTokenController).Methods(http.MethodPost) @@ -52,10 +52,10 @@ func (s *Server) initializeRoutes() { messageRepository := repositories.NewMessageRepository(s.DB) messageService := services.NewMessageService(s.logger, messageRepository, chatSessionRepository, s.embedder, s.pineconeVectorDB, s.openAIClient) - createChatSessionHandler := chatSessions.NewCreateUserChatSessionHandler(chatSessionService, s.logger) - getChatSessionHandler := chatSessions.NewGetChatSessionHandler(chatSessionService, s.logger) - sendMessageHandler := chatSessions.NewSendMessageHandler(messageService, s.logger) - submitFeedbackHandler := chatSessions.NewSubmitFeedbackHandler(messageService, s.logger) + createChatSessionHandler := chatSessions2.NewCreateUserChatSessionHandler(chatSessionService, s.logger) + getChatSessionHandler := chatSessions2.NewGetChatSessionHandler(chatSessionService, s.logger) + sendMessageHandler := chatSessions2.NewSendMessageHandler(messageService, s.logger) + submitFeedbackHandler := chatSessions2.NewSubmitFeedbackHandler(messageService, s.logger) protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionController).Methods("POST") protected.HandleFunc("/users/{user_id}/chat-sessions", getChatSessionHandler.GetUserChatSessionsController).Methods("GET") diff --git a/pkg/server/server.go b/pkg/server/http/server.go similarity index 99% rename from pkg/server/server.go rename to pkg/server/http/server.go index 0329424..e08b3a1 100644 --- a/pkg/server/server.go +++ b/pkg/server/http/server.go @@ -1,4 +1,4 @@ -package server +package http import ( "context" From 3ded284c7ef9c35db037dfdc522267a4c01a5dbd Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Sun, 13 Jul 2025 12:28:57 +0300 Subject: [PATCH 46/52] Adds dummy mcp server (not working) --- cmd/http/main.go | 9 +- go.mod | 7 +- go.sum | 39 ++---- internal/handlers/mcp/addHandler.go | 1 + pkg/server/http/routes.go | 4 + pkg/server/http/server.go | 4 + pkg/server/mcp/server.go | 17 +++ pkg/server/mcp/tools.go | 182 ++++++++++++++++++++++++++++ 8 files changed, 232 insertions(+), 31 deletions(-) create mode 100644 internal/handlers/mcp/addHandler.go create mode 100644 pkg/server/mcp/server.go create mode 100644 pkg/server/mcp/tools.go diff --git a/cmd/http/main.go b/cmd/http/main.go index 11887e2..7512d69 100644 --- a/cmd/http/main.go +++ b/cmd/http/main.go @@ -11,7 +11,9 @@ import ( "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" "github.com/loukaspe/jedi-team-challenge/pkg/logger" http2 "github.com/loukaspe/jedi-team-challenge/pkg/server/http" + "github.com/loukaspe/jedi-team-challenge/pkg/server/mcp" "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" + "github.com/mark3labs/mcp-go/server" "github.com/openai/openai-go" "github.com/openai/openai-go/option" "github.com/pinecone-io/go-pinecone/v3/pinecone" @@ -44,7 +46,12 @@ func main() { } db := getDB() - server := http2.NewServer(db, router, httpServer, logger, &client, embedder, pineconeVectorDB) + mcpServer := mcp.NewServer(server.NewMCPServer( + os.Getenv("MCP_SERVER_NAME"), + os.Getenv("MCP_SERVER_VERSION"), + )) + + server := http2.NewServer(db, router, httpServer, mcpServer, logger, &client, embedder, pineconeVectorDB) server.Run() } diff --git a/go.mod b/go.mod index 0b42df0..a54a17e 100644 --- a/go.mod +++ b/go.mod @@ -7,10 +7,10 @@ toolchain go1.23.1 require ( github.com/DATA-DOG/go-sqlmock v1.5.2 github.com/golang-jwt/jwt/v4 v4.5.0 - github.com/golang/mock v1.6.0 github.com/google/uuid v1.6.0 github.com/gorilla/mux v1.8.0 github.com/joho/godotenv v1.5.1 + github.com/mark3labs/mcp-go v0.32.0 github.com/openai/openai-go v1.1.0 github.com/pinecone-io/go-pinecone/v3 v3.1.0 github.com/pkoukk/tiktoken-go v0.1.7 @@ -19,6 +19,7 @@ require ( github.com/swaggo/swag v1.16.4 go.uber.org/mock v0.5.2 golang.org/x/crypto v0.32.0 + google.golang.org/protobuf v1.34.1 gorm.io/driver/postgres v1.5.11 gorm.io/gorm v1.25.10 ) @@ -44,11 +45,12 @@ require ( github.com/mailru/easyjson v0.7.7 // indirect github.com/oapi-codegen/runtime v1.1.1 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect - github.com/rogpeppe/go-internal v1.6.1 // indirect + github.com/spf13/cast v1.7.1 // indirect github.com/tidwall/gjson v1.14.4 // indirect github.com/tidwall/match v1.1.1 // indirect github.com/tidwall/pretty v1.2.1 // indirect github.com/tidwall/sjson v1.2.5 // indirect + github.com/yosida95/uritemplate/v3 v3.0.2 // indirect golang.org/x/net v0.34.0 // indirect golang.org/x/sync v0.11.0 // indirect golang.org/x/sys v0.30.0 // indirect @@ -57,7 +59,6 @@ require ( google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157 // indirect google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157 // indirect google.golang.org/grpc v1.65.0 // indirect - google.golang.org/protobuf v1.34.1 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index ab85064..1ca6b30 100644 --- a/go.sum +++ b/go.sum @@ -16,6 +16,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/dlclark/regexp2 v1.10.0 h1:+/GIL799phkJqYW+3YbOd8LCcbHzT0Pbo8zl70MHsq0= github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= +github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= +github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= github.com/go-openapi/jsonpointer v0.19.5 h1:gZr+CIYByUqjcgeLXnQu2gHYQC9o73G2XUeOFYEICuY= github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= @@ -28,8 +30,6 @@ github.com/go-openapi/swag v0.19.15 h1:D2NRCBzS9/pEY3gP9Nl8aDqGUcPFrwG2p+CNFrLyr github.com/go-openapi/swag v0.19.15/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ= github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg= github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= -github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc= -github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= @@ -55,8 +55,8 @@ github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFF github.com/juju/gnuflag v0.0.0-20171113085948-2ce1bb71843d/go.mod h1:2PavIy+JPciBPrBUjwbNvtwB6RQlve+hkpll6QSNmOE= github.com/kisielk/sqlstruct v0.0.0-20201105191214-5f3e10d3ab46/go.mod h1:yyMNCyc/Ib3bDTKd379tNMpB/7/H5TjM2Y9QJ5THLbE= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= -github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= +github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= @@ -66,6 +66,8 @@ github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN github.com/mailru/easyjson v0.7.6/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= +github.com/mark3labs/mcp-go v0.32.0 h1:fgwmbfL2gbd67obg57OfV2Dnrhs1HtSdlY/i5fn7MU8= +github.com/mark3labs/mcp-go v0.32.0/go.mod h1:rXqOudj/djTORU/ThxYx8fqEVj/5pvTuuebQ2RC7uk4= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/oapi-codegen/runtime v1.1.1 h1:EXLHh0DXIJnWhdRPN2w4MXAzFyE4CskzhNLUmtpMYro= github.com/oapi-codegen/runtime v1.1.1/go.mod h1:SK9X900oXmPWilYR5/WKPzt3Kqxn/uS/+lbpREv+eCg= @@ -77,10 +79,12 @@ github.com/pkoukk/tiktoken-go v0.1.7 h1:qOBHXX4PHtvIvmOtyg1EeKlwFRiMKAcoMp4Q+bLQ github.com/pkoukk/tiktoken-go v0.1.7/go.mod h1:9NiV+i9mJKGj1rYOT+njbv+ZwA/zJxYdewGl6qVatpg= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/rogpeppe/go-internal v1.6.1 h1:/FiVV8dS/e+YqF2JvO3yXRFbBLTIuSDkuC7aBOAvL+k= -github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= +github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= +github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +github.com/spf13/cast v1.7.1 h1:cuNEagBQEHWN1FnbGEjCXL2szYEXqfJPbP2HNUaca9Y= +github.com/spf13/cast v1.7.1/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= github.com/spkg/bom v0.0.0-20160624110644-59b7046e48ad/go.mod h1:qLr4V1qq6nMqFKkMo8ZTx3f+BZEkzsRUY10Xsm2mwU0= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= @@ -100,50 +104,32 @@ github.com/tidwall/pretty v1.2.1 h1:qjsOFOWWQl+N3RsoF5/ssm1pHmJJwhjlSbZ51I6wMl4= github.com/tidwall/pretty v1.2.1/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY= github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28= -github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +github.com/yosida95/uritemplate/v3 v3.0.2 h1:Ed3Oyj9yrmi9087+NczuL5BwkIc4wvTb5zIM+UJPGz4= +github.com/yosida95/uritemplate/v3 v3.0.2/go.mod h1:ILOh0sOhIJR3+L/8afwt/kE++YT040gmv5BQTMR2HP4= go.uber.org/mock v0.5.2 h1:LbtPTcP8A5k9WPXj54PPPbjcI4Y6lhyOZXn+VS7wNko= go.uber.org/mock v0.5.2/go.mod h1:wLlUxC2vVTPTaE3UD51E0BGOAElKrILxhVSDYQLld5o= -golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= -golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.32.0 h1:euUpcYgM8WcP71gNpTqQCn6rC2t6ULUPiOzfWaXVVfc= golang.org/x/crypto v0.32.0/go.mod h1:ZnnJkOaASj8g0AjIduWNlq2NRxL0PlBrbKVyZ6V/Ugc= -golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.18.0 h1:5+9lSbEzPSdWkH32vYPBwEpX8KwDbM52Ud9xBUvNlb0= golang.org/x/mod v0.18.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= -golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= -golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= golang.org/x/net v0.0.0-20210421230115-4e50805a0758/go.mod h1:72T/g9IO56b78aLF+1Kcs5dz7/ng1VjMUvfKvpfy+jM= golang.org/x/net v0.34.0 h1:Mb7Mrk043xzHgnRM88suvJFwzVrRfHEHJEl5/71CKw0= golang.org/x/net v0.34.0/go.mod h1:di0qlW3YNM5oh6GqDGQr92MyTozJPmybPK4Ev/Gm31k= -golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.11.0 h1:GGz8+XQP4FvTTrjZPzNKTMFtSXH80RAzG+5ghFPgK9w= golang.org/x/sync v0.11.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= -golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210420072515-93ed5bcd2bfe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc= golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= -golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.22.0 h1:bofq7m3/HAFvbF51jz3Q9wLg3jkvSPuiZu/pD1XwgtM= golang.org/x/text v0.22.0/go.mod h1:YRoo4H8PVmsu+E3Ou7cqLVH8oXWIHVoX0jqUWALQhfY= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= -golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.22.0 h1:gqSGLZqv+AI9lIQzniJ0nZDRG5GBPsSi+DRNHWNz6yA= golang.org/x/tools v0.22.0/go.mod h1:aCwcsjqvq7Yqt6TNyX7QMU2enbQ/Gt0bo6krSeEri+c= -golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157 h1:7whR9kGa5LUwFtpLm2ArCEejtnxlGeLbAyjFY8sGNFw= google.golang.org/genproto/googleapis/api v0.0.0-20240528184218-531527333157/go.mod h1:99sLkeliLXfdj2J75X3Ho+rrVCaJze0uwN7zDDkjPVU= google.golang.org/genproto/googleapis/rpc v0.0.0-20240528184218-531527333157 h1:Zy9XzmMEflZ/MAaA7vNcoebnRAld7FsPW1EeBB7V0m8= @@ -157,7 +143,6 @@ gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8 gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= -gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= diff --git a/internal/handlers/mcp/addHandler.go b/internal/handlers/mcp/addHandler.go new file mode 100644 index 0000000..87468eb --- /dev/null +++ b/internal/handlers/mcp/addHandler.go @@ -0,0 +1 @@ +package mcp diff --git a/pkg/server/http/routes.go b/pkg/server/http/routes.go index 0aa02d6..23dbba3 100644 --- a/pkg/server/http/routes.go +++ b/pkg/server/http/routes.go @@ -33,6 +33,10 @@ func (s *Server) initializeRoutes() { healthCheckHandler := http2.NewHealthCheckHandler(s.DB) s.router.HandleFunc("/health-check", healthCheckHandler.HealthCheckController).Methods("GET") + mcpSSEServer := s.mcpServer.InitialiseSSEServer() + + s.router.HandleFunc("/mcp", mcpSSEServer.ServeHTTP) + // auth jwtMechanism := auth.NewAuthMechanism( os.Getenv("JWT_SECRET_KEY"), diff --git a/pkg/server/http/server.go b/pkg/server/http/server.go index e08b3a1..081e51f 100644 --- a/pkg/server/http/server.go +++ b/pkg/server/http/server.go @@ -6,6 +6,7 @@ import ( "github.com/gorilla/mux" "github.com/loukaspe/jedi-team-challenge/pkg/embeddings" "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "github.com/loukaspe/jedi-team-challenge/pkg/server/mcp" "github.com/loukaspe/jedi-team-challenge/pkg/vectordb" "github.com/openai/openai-go" log "github.com/sirupsen/logrus" @@ -20,6 +21,7 @@ import ( type Server struct { DB *gorm.DB httpServer *http.Server + mcpServer *mcp.Server router *mux.Router logger logger.LoggerInterface openAIClient *openai.Client @@ -31,6 +33,7 @@ func NewServer( db *gorm.DB, router *mux.Router, httpServer *http.Server, + mcpServer *mcp.Server, logger logger.LoggerInterface, openAIClient *openai.Client, embedder *embeddings.EmbeddingService, @@ -40,6 +43,7 @@ func NewServer( DB: db, router: router, httpServer: httpServer, + mcpServer: mcpServer, logger: logger, openAIClient: openAIClient, embedder: embedder, diff --git a/pkg/server/mcp/server.go b/pkg/server/mcp/server.go new file mode 100644 index 0000000..e9b5b88 --- /dev/null +++ b/pkg/server/mcp/server.go @@ -0,0 +1,17 @@ +package mcp + +import ( + "github.com/mark3labs/mcp-go/server" +) + +type Server struct { + mcpServer *server.MCPServer +} + +func NewServer( + mcpServer *server.MCPServer, +) *Server { + return &Server{ + mcpServer: mcpServer, + } +} diff --git a/pkg/server/mcp/tools.go b/pkg/server/mcp/tools.go new file mode 100644 index 0000000..37fe0a7 --- /dev/null +++ b/pkg/server/mcp/tools.go @@ -0,0 +1,182 @@ +package mcp + +import ( + "context" + "errors" + "fmt" + "github.com/mark3labs/mcp-go/mcp" + "github.com/mark3labs/mcp-go/server" +) + +func (s *Server) InitialiseSSEServer() *server.SSEServer { + add := mcp.NewTool("add", + mcp.WithDescription("Add two numbers"), + mcp.WithNumber("a", + mcp.Required(), + mcp.Description("First number"), + ), + mcp.WithNumber("b", + mcp.Required(), + mcp.Description("Second number"), + ), + ) + + // Add subtraction tool + subtract := mcp.NewTool("subtract", + mcp.WithDescription("Subtract second number from first number"), + mcp.WithNumber("a", + mcp.Required(), + mcp.Description("First number"), + ), + mcp.WithNumber("b", + mcp.Required(), + mcp.Description("Second number"), + ), + ) + + // Add multiplication tool + multiply := mcp.NewTool("multiply", + mcp.WithDescription("Multiply two numbers"), + mcp.WithNumber("a", + mcp.Required(), + mcp.Description("First number"), + ), + mcp.WithNumber("b", + mcp.Required(), + mcp.Description("Second number"), + ), + ) + + // Add division tool + divide := mcp.NewTool("divide", + mcp.WithDescription("Divide first number by second number"), + mcp.WithNumber("a", + mcp.Required(), + mcp.Description("Numerator"), + ), + mcp.WithNumber("b", + mcp.Required(), + mcp.Description("Denominator"), + ), + ) + + // Add percentage tool + percentage := mcp.NewTool("percentage", + mcp.WithDescription("Calculate what percentage the first number is of the second number"), + mcp.WithNumber("a", + mcp.Required(), + mcp.Description("Part value"), + ), + mcp.WithNumber("b", + mcp.Required(), + mcp.Description("Total value"), + ), + ) + + // Add tool handlers + s.mcpServer.AddTool(add, addHandler) + s.mcpServer.AddTool(subtract, subtractHandler) + s.mcpServer.AddTool(multiply, multiplyHandler) + s.mcpServer.AddTool(divide, divideHandler) + s.mcpServer.AddTool(percentage, percentageHandler) + + return server.NewSSEServer( + s.mcpServer, + server.WithStaticBasePath("/"), + server.WithSSEEndpoint("/sse"), + server.WithMessageEndpoint("/message"), + ) +} + +// addHandler processes addition requests, adding two numbers and returning the formatted result +// Returns an error if inputs are not valid numbers +func addHandler(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) { + a, err := request.RequireFloat("a") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + b, err := request.RequireFloat("b") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + result := a + b + return mcp.NewToolResultText(fmt.Sprintf("%.2f", result)), nil +} + +// subtractHandler processes subtraction requests, subtracting the second number from the first +// Returns an error if inputs are not valid numbers +func subtractHandler(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) { + a, err := request.RequireFloat("a") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + b, err := request.RequireFloat("b") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + result := a - b + return mcp.NewToolResultText(fmt.Sprintf("%.2f", result)), nil +} + +// multiplyHandler processes multiplication requests, multiplying two numbers together +// Returns an error if inputs are not valid numbers +func multiplyHandler(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) { + a, err := request.RequireFloat("a") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + b, err := request.RequireFloat("b") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + result := a * b + return mcp.NewToolResultText(fmt.Sprintf("%.2f", result)), nil +} + +// divideHandler processes division requests, dividing the first number by the second +// Returns an error if inputs are not valid numbers or if dividing by zero +func divideHandler(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) { + a, err := request.RequireFloat("a") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + b, err := request.RequireFloat("b") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + if b == 0 { + return nil, errors.New("cannot divide by zero") + } + + result := a / b + return mcp.NewToolResultText(fmt.Sprintf("%.2f", result)), nil +} + +// percentageHandler calculates what percentage the first number is of the second number +// Returns an error if inputs are not valid numbers or if the total value is zero +func percentageHandler(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) { + a, err := request.RequireFloat("a") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + b, err := request.RequireFloat("b") + if err != nil { + return mcp.NewToolResultError(err.Error()), nil + } + + if b == 0 { + return nil, errors.New("total value cannot be zero") + } + + result := (a / b) * 100 + return mcp.NewToolResultText(fmt.Sprintf("%.2f%%", result)), nil +} From eb925c4e2dba53a12e2bc0a8779f740eeb96269c Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Fri, 18 Jul 2025 14:21:54 +0300 Subject: [PATCH 47/52] Small change --- cmd/{http => }/main.go | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename cmd/{http => }/main.go (100%) diff --git a/cmd/http/main.go b/cmd/main.go similarity index 100% rename from cmd/http/main.go rename to cmd/main.go From 2cf30f8afc1b27eeefef4e1fbe0fc53485c86cdf Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Fri, 18 Jul 2025 15:11:01 +0300 Subject: [PATCH 48/52] Adds dummy ws routes --- build/Dev.Dockerfile | 2 +- build/Dockerfile | 2 +- go.mod | 1 + go.sum | 2 + internal/handlers/http/addHandler.go | 57 +++++++++++++++++++++++ internal/handlers/http/subtractHandler.go | 52 +++++++++++++++++++++ pkg/server/http/routes.go | 30 ++++++++---- 7 files changed, 135 insertions(+), 11 deletions(-) create mode 100644 internal/handlers/http/addHandler.go create mode 100644 internal/handlers/http/subtractHandler.go diff --git a/build/Dev.Dockerfile b/build/Dev.Dockerfile index 04a525f..de9fe27 100755 --- a/build/Dev.Dockerfile +++ b/build/Dev.Dockerfile @@ -18,7 +18,7 @@ WORKDIR /app COPY .. . -RUN GOOS=linux go build -gcflags='all=-N -l' -tags musl -a -installsuffix cgo -o main ./cmd/http/main.go +RUN GOOS=linux go build -gcflags='all=-N -l' -tags musl -a -installsuffix cgo -o main ./cmd/main.go EXPOSE 8080 EXPOSE 40000 diff --git a/build/Dockerfile b/build/Dockerfile index 1054377..04f4f6c 100755 --- a/build/Dockerfile +++ b/build/Dockerfile @@ -18,7 +18,7 @@ RUN go mod download COPY .. . # Build the Go app -RUN GO111MODULE=on CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o main ./cmd/http/main.go +RUN GO111MODULE=on CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o main ./cmd/main.go # Start a new stage from scratch FROM alpine:latest diff --git a/go.mod b/go.mod index a54a17e..b493dbd 100644 --- a/go.mod +++ b/go.mod @@ -9,6 +9,7 @@ require ( github.com/golang-jwt/jwt/v4 v4.5.0 github.com/google/uuid v1.6.0 github.com/gorilla/mux v1.8.0 + github.com/gorilla/websocket v1.5.3 github.com/joho/godotenv v1.5.1 github.com/mark3labs/mcp-go v0.32.0 github.com/openai/openai-go v1.1.0 diff --git a/go.sum b/go.sum index 1ca6b30..1d594b8 100644 --- a/go.sum +++ b/go.sum @@ -36,6 +36,8 @@ github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= +github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg= +github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk= diff --git a/internal/handlers/http/addHandler.go b/internal/handlers/http/addHandler.go new file mode 100644 index 0000000..e63bd81 --- /dev/null +++ b/internal/handlers/http/addHandler.go @@ -0,0 +1,57 @@ +package http + +import ( + "encoding/json" + "fmt" + "github.com/gorilla/websocket" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" +) + +type Payload struct { + A float64 `json:"a"` + B float64 `json:"b"` +} + +type AddHandler struct { + logger logger.LoggerInterface + upgrader *websocket.Upgrader +} + +func NewAddHandler( + upgrader *websocket.Upgrader, + logger logger.LoggerInterface, +) *AddHandler { + return &AddHandler{ + upgrader: upgrader, + logger: logger, + } +} + +func (handler AddHandler) AddController(w http.ResponseWriter, r *http.Request) { + conn, err := handler.upgrader.Upgrade(w, r, nil) + if err != nil { + handler.logger.Error("Upgrade error:", map[string]interface{}{ + "errorMessage": err.Error(), + }) + return + } + defer conn.Close() + + for { + _, msg, err := conn.ReadMessage() + if err != nil { + fmt.Println("Read error:", err) + break + } + + var data Payload + if err := json.Unmarshal(msg, &data); err != nil { + conn.WriteMessage(websocket.TextMessage, []byte("Invalid input")) + continue + } + + result := data.A + data.B + conn.WriteMessage(websocket.TextMessage, []byte(fmt.Sprintf("Result: %.2f", result))) + } +} diff --git a/internal/handlers/http/subtractHandler.go b/internal/handlers/http/subtractHandler.go new file mode 100644 index 0000000..caffaa6 --- /dev/null +++ b/internal/handlers/http/subtractHandler.go @@ -0,0 +1,52 @@ +package http + +import ( + "encoding/json" + "fmt" + "github.com/gorilla/websocket" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" +) + +type SubtractHandler struct { + logger logger.LoggerInterface + upgrader *websocket.Upgrader +} + +func NewSubtractHandler( + upgrader *websocket.Upgrader, + logger logger.LoggerInterface, +) *SubtractHandler { + return &SubtractHandler{ + upgrader: upgrader, + logger: logger, + } +} + +func (handler SubtractHandler) SubtractController(w http.ResponseWriter, r *http.Request) { + conn, err := handler.upgrader.Upgrade(w, r, nil) + if err != nil { + handler.logger.Error("Upgrade error:", map[string]interface{}{ + "errorMessage": err.Error(), + }) + return + } + defer conn.Close() + + for { + _, msg, err := conn.ReadMessage() + if err != nil { + fmt.Println("Read error:", err) + break + } + + var data Payload + if err := json.Unmarshal(msg, &data); err != nil { + conn.WriteMessage(websocket.TextMessage, []byte("Invalid input")) + continue + } + + result := data.A - data.B + conn.WriteMessage(websocket.TextMessage, []byte(fmt.Sprintf("Result: %.2f", result))) + } +} diff --git a/pkg/server/http/routes.go b/pkg/server/http/routes.go index 23dbba3..8c8e6f8 100644 --- a/pkg/server/http/routes.go +++ b/pkg/server/http/routes.go @@ -1,9 +1,10 @@ package http import ( + "github.com/gorilla/websocket" "github.com/loukaspe/jedi-team-challenge/internal/core/services" - http2 "github.com/loukaspe/jedi-team-challenge/internal/handlers/http" - chatSessions2 "github.com/loukaspe/jedi-team-challenge/internal/handlers/http/chatSessions" + httpHandlers "github.com/loukaspe/jedi-team-challenge/internal/handlers/http" + chatSessionHandlers "github.com/loukaspe/jedi-team-challenge/internal/handlers/http/chatSessions" "github.com/loukaspe/jedi-team-challenge/internal/repositories" "github.com/loukaspe/jedi-team-challenge/pkg/auth" @@ -29,8 +30,17 @@ import ( // @accept json // @produce json func (s *Server) initializeRoutes() { + var upgrader = websocket.Upgrader{ + CheckOrigin: func(r *http.Request) bool { + return true // Allow all origins for demo + }, + } + + wsAddHandler := httpHandlers.NewAddHandler(&upgrader, s.logger) + wsSubtractHandler := httpHandlers.NewSubtractHandler(&upgrader, s.logger) + // health check - healthCheckHandler := http2.NewHealthCheckHandler(s.DB) + healthCheckHandler := httpHandlers.NewHealthCheckHandler(s.DB) s.router.HandleFunc("/health-check", healthCheckHandler.HealthCheckController).Methods("GET") mcpSSEServer := s.mcpServer.InitialiseSSEServer() @@ -43,8 +53,8 @@ func (s *Server) initializeRoutes() { os.Getenv("JWT_SIGNING_METHOD"), ) jwtService := services.NewJwtService(jwtMechanism) - jwtMiddleware := http2.NewAuthenticationMw(jwtMechanism) - jwtHandler := http2.NewJwtClaimsHandler(jwtService, s.logger) + jwtMiddleware := httpHandlers.NewAuthenticationMw(jwtMechanism) + jwtHandler := httpHandlers.NewJwtClaimsHandler(jwtService, s.logger) s.router.HandleFunc("/token", jwtHandler.JwtTokenController).Methods(http.MethodPost) @@ -56,10 +66,10 @@ func (s *Server) initializeRoutes() { messageRepository := repositories.NewMessageRepository(s.DB) messageService := services.NewMessageService(s.logger, messageRepository, chatSessionRepository, s.embedder, s.pineconeVectorDB, s.openAIClient) - createChatSessionHandler := chatSessions2.NewCreateUserChatSessionHandler(chatSessionService, s.logger) - getChatSessionHandler := chatSessions2.NewGetChatSessionHandler(chatSessionService, s.logger) - sendMessageHandler := chatSessions2.NewSendMessageHandler(messageService, s.logger) - submitFeedbackHandler := chatSessions2.NewSubmitFeedbackHandler(messageService, s.logger) + createChatSessionHandler := chatSessionHandlers.NewCreateUserChatSessionHandler(chatSessionService, s.logger) + getChatSessionHandler := chatSessionHandlers.NewGetChatSessionHandler(chatSessionService, s.logger) + sendMessageHandler := chatSessionHandlers.NewSendMessageHandler(messageService, s.logger) + submitFeedbackHandler := chatSessionHandlers.NewSubmitFeedbackHandler(messageService, s.logger) protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionController).Methods("POST") protected.HandleFunc("/users/{user_id}/chat-sessions", getChatSessionHandler.GetUserChatSessionsController).Methods("GET") @@ -68,4 +78,6 @@ func (s *Server) initializeRoutes() { protected.HandleFunc("/chat-sessions/{session_id}", getChatSessionHandler.GetChatSessionController).Methods("GET") + protected.HandleFunc("/ws/add", wsAddHandler.AddController) + protected.HandleFunc("/ws/subtract", wsSubtractHandler.SubtractController) } From 801516e15a94926cc8f2fa82acb6307eac42a046 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Fri, 18 Jul 2025 16:03:53 +0300 Subject: [PATCH 49/52] Create send message in websocket --- .../http/chatSessions/wsSendMessageHandler.go | 120 ++++++++++++++++++ pkg/server/http/routes.go | 2 + script/e2e.sh | 23 +++- 3 files changed, 144 insertions(+), 1 deletion(-) create mode 100644 internal/handlers/http/chatSessions/wsSendMessageHandler.go diff --git a/internal/handlers/http/chatSessions/wsSendMessageHandler.go b/internal/handlers/http/chatSessions/wsSendMessageHandler.go new file mode 100644 index 0000000..8e0d4e2 --- /dev/null +++ b/internal/handlers/http/chatSessions/wsSendMessageHandler.go @@ -0,0 +1,120 @@ +package chatSessions + +import ( + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/gorilla/websocket" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + "github.com/loukaspe/jedi-team-challenge/internal/repositories" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" +) + +type WsSendMessageHandler struct { + MessageService services.MessageServiceInterface + logger logger.LoggerInterface + upgrader *websocket.Upgrader +} + +func NewWsSendMessageHandler( + service services.MessageServiceInterface, + logger logger.LoggerInterface, + upgrader *websocket.Upgrader, +) *WsSendMessageHandler { + return &WsSendMessageHandler{ + MessageService: service, + logger: logger, + upgrader: upgrader, + } +} + +// Request model from WebSocket client +type SendMessageWSRequest struct { + Content string `json:"content"` +} + +// WebSocket message handler +func (handler *WsSendMessageHandler) WsSendMessageController(w http.ResponseWriter, r *http.Request) { + userIdStr := mux.Vars(r)["user_id"] + if userIdStr == "" { + http.Error(w, "missing user id", http.StatusBadRequest) + return + } + userId, err := uuid.Parse(userIdStr) + if err != nil { + http.Error(w, "malformed user uuid", http.StatusBadRequest) + return + } + + sessionIdStr := mux.Vars(r)["session_id"] + if sessionIdStr == "" { + http.Error(w, "missing session id", http.StatusBadRequest) + return + } + chatSessionID, err := uuid.Parse(sessionIdStr) + if err != nil { + http.Error(w, "malformed session uuid", http.StatusBadRequest) + return + } + + conn, err := handler.upgrader.Upgrade(w, r, nil) + if err != nil { + handler.logger.Error("Upgrade error", map[string]interface{}{"errorMessage": err.Error()}) + return + } + defer conn.Close() + + for { + var request SendMessageWSRequest + if err := conn.ReadJSON(&request); err != nil { + handler.logger.Error("Invalid WS JSON input", map[string]interface{}{"errorMessage": err.Error()}) + conn.WriteJSON(SendMessageResponse{ErrorMessage: "Invalid input format"}) + return + } + + domainMessage := &domain.Message{ + ChatSessionID: chatSessionID, + Content: request.Content, + Sender: repositories.USER_SENDER, + } + + insertedUUID, err := handler.MessageService.CreateMessage(r.Context(), userId, domainMessage) + if notFound, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Chat session not found", map[string]interface{}{"errorMessage": notFound.Unwrap()}) + conn.WriteJSON(SendMessageResponse{ErrorMessage: "chat session not found: " + notFound.Error()}) + continue + } + if mismatch, ok := err.(customerrors.UserMismatchError); ok { + handler.logger.Error("User mismatch error", map[string]interface{}{"errorMessage": mismatch.Error()}) + conn.WriteJSON(SendMessageResponse{ErrorMessage: "user mismatch: " + mismatch.Error()}) + continue + } + if err != nil { + handler.logger.Error("Error creating message", map[string]interface{}{"errorMessage": err.Error()}) + conn.WriteJSON(SendMessageResponse{ErrorMessage: "internal error: " + err.Error()}) + continue + } + + domainMessage.ID = insertedUUID + + replyMessage, err := handler.MessageService.GetAnswerForMessage(r.Context(), insertedUUID) + if notFound, ok := err.(customerrors.ResourceNotFoundErrorWrapper); ok { + handler.logger.Error("Reply not found", map[string]interface{}{"errorMessage": notFound.Unwrap()}) + conn.WriteJSON(SendMessageResponse{ErrorMessage: notFound.Error()}) + continue + } + if err != nil { + handler.logger.Error("Error fetching reply", map[string]interface{}{"errorMessage": err.Error()}) + conn.WriteJSON(SendMessageResponse{ErrorMessage: "failed to fetch reply"}) + continue + } + + // Send back both user and system messages + conn.WriteJSON(SendMessageResponse{ + UserMessage: MessageResponseFromModel(domainMessage), + SystemMessage: MessageResponseFromModel(replyMessage), + }) + } +} diff --git a/pkg/server/http/routes.go b/pkg/server/http/routes.go index 8c8e6f8..b9c9968 100644 --- a/pkg/server/http/routes.go +++ b/pkg/server/http/routes.go @@ -69,6 +69,7 @@ func (s *Server) initializeRoutes() { createChatSessionHandler := chatSessionHandlers.NewCreateUserChatSessionHandler(chatSessionService, s.logger) getChatSessionHandler := chatSessionHandlers.NewGetChatSessionHandler(chatSessionService, s.logger) sendMessageHandler := chatSessionHandlers.NewSendMessageHandler(messageService, s.logger) + wsSendMessageHandler := chatSessionHandlers.NewWsSendMessageHandler(messageService, s.logger, &upgrader) submitFeedbackHandler := chatSessionHandlers.NewSubmitFeedbackHandler(messageService, s.logger) protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionController).Methods("POST") @@ -80,4 +81,5 @@ func (s *Server) initializeRoutes() { protected.HandleFunc("/ws/add", wsAddHandler.AddController) protected.HandleFunc("/ws/subtract", wsSubtractHandler.SubtractController) + protected.HandleFunc("/ws/users/{user_id}/chat-sessions/{session_id}/messages", wsSendMessageHandler.WsSendMessageController) } diff --git a/script/e2e.sh b/script/e2e.sh index a46b093..028174a 100644 --- a/script/e2e.sh +++ b/script/e2e.sh @@ -59,6 +59,28 @@ system_message_id_3=$(echo "$response_message_3" | jq -r '.systemMessage.id') echo "Three messages were sent in the same discussion and title was generated" echo "----------------------------------" +# Step 5.5: Send three messages with websockets + +messages=( + "what do you know about australian millennial parents" + "do they use social media" + "what social media do they use the most" +) + +echo "Connecting to WebSocket..." +for msg in "${messages[@]}"; do + echo "Sending message: $msg" + + # Send JSON and read one response line + response=$(echo "{\"content\":\"$msg\"}" | + websocat --header="Authorization: Bearer $token" --text "ws://localhost:8080/ws/users/$USER_ID/chat-sessions/$CHAT_SESSION_ID/messages") + + echo "Received response:" + echo "$response" +done + +echo "----------------------------------" + # Step 6: Fetch and display the whole chat session echo "Fetching the entire chat session $CHAT_SESSION_ID..." chat_session_response=$(curl -s --location "$BASE_URL/chat-sessions/$CHAT_SESSION_ID" \ @@ -91,6 +113,5 @@ response_message_4=$(curl -s --location "$BASE_URL/users/$USER_ID/chat-sessions/ --header "Authorization: Bearer $token" \ --data "{\"content\":\"$message_content_4\"}") - echo "Message 4 that's supposed to be not answered sent to chat session $CHAT_SESSION_ID:" echo "$response_message_4" From 3d30cd224905c0e11598492356c9de749ca70847 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Fri, 18 Jul 2025 16:40:52 +0300 Subject: [PATCH 50/52] Adds sse streaming send message --- internal/core/domain/chatSession.go | 13 ++ internal/core/services/messageService.go | 33 +++++ .../chatSessions/sseSendMessageHandler.go | 124 ++++++++++++++++++ pkg/server/http/routes.go | 2 + 4 files changed, 172 insertions(+) create mode 100644 internal/handlers/http/chatSessions/sseSendMessageHandler.go diff --git a/internal/core/domain/chatSession.go b/internal/core/domain/chatSession.go index f581757..eb4866b 100644 --- a/internal/core/domain/chatSession.go +++ b/internal/core/domain/chatSession.go @@ -1,6 +1,7 @@ package domain import ( + "fmt" "github.com/google/uuid" "time" ) @@ -22,3 +23,15 @@ type Message struct { CreatedAt time.Time Feedback *string } + +func (m Message) String() string { + return fmt.Sprintf( + "Message(ID: %s, ChatSessionID: %s, Sender: %s, Content: %s, CreatedAt: %s, Feedback: %v)", + m.ID.String(), + m.ChatSessionID.String(), + m.Sender, + m.Content, + m.CreatedAt.Format(time.RFC3339), + m.Feedback, + ) +} diff --git a/internal/core/services/messageService.go b/internal/core/services/messageService.go index c1d3892..034b758 100644 --- a/internal/core/services/messageService.go +++ b/internal/core/services/messageService.go @@ -13,11 +13,13 @@ import ( "github.com/loukaspe/jedi-team-challenge/pkg/logger" "github.com/openai/openai-go" "strings" + "time" ) type MessageServiceInterface interface { CreateMessage(context.Context, uuid.UUID, *domain.Message) (uuid.UUID, error) GetAnswerForMessage(context.Context, uuid.UUID) (*domain.Message, error) + StreamAnswerForMessage(context.Context, uuid.UUID) (<-chan string, <-chan error) UpdateMessageFeedback(ctx context.Context, message *domain.Message, userID uuid.UUID) error } @@ -146,6 +148,37 @@ func (s *MessageService) GetAnswerForMessage(ctx context.Context, initialMessage return replyMessage, nil } +func (s *MessageService) StreamAnswerForMessage(ctx context.Context, initialMessageID uuid.UUID) (<-chan string, <-chan error) { + tokenChan := make(chan string) + errChan := make(chan error, 1) + + go func() { + defer close(tokenChan) + defer close(errChan) + + response, err := s.GetAnswerForMessage(ctx, initialMessageID) + if err != nil { + errChan <- err + return + } + + // split into tokens (simulate streaming) + tokens := strings.Split(response.String(), " ") + + for _, token := range tokens { + select { + case <-ctx.Done(): + errChan <- ctx.Err() + return + case tokenChan <- token: + time.Sleep(100 * time.Millisecond) // simulate generation delay + } + } + }() + + return tokenChan, errChan +} + func (s *MessageService) generateAnswerFromOpenAI(ctx context.Context, text []string, initialMessage string, previousMessages []*domain.Message) (string, error) { prompt := fmt.Sprintf(`Use the following context to answer the question. Context: diff --git a/internal/handlers/http/chatSessions/sseSendMessageHandler.go b/internal/handlers/http/chatSessions/sseSendMessageHandler.go new file mode 100644 index 0000000..7a414a1 --- /dev/null +++ b/internal/handlers/http/chatSessions/sseSendMessageHandler.go @@ -0,0 +1,124 @@ +package chatSessions + +import ( + "encoding/json" + "fmt" + "github.com/google/uuid" + "github.com/gorilla/mux" + "github.com/loukaspe/jedi-team-challenge/internal/core/domain" + "github.com/loukaspe/jedi-team-challenge/internal/core/services" + "github.com/loukaspe/jedi-team-challenge/internal/repositories" + customerrors "github.com/loukaspe/jedi-team-challenge/pkg/errors" + "github.com/loukaspe/jedi-team-challenge/pkg/logger" + "net/http" + "strings" +) + +type SseSendMessageHandler struct { + MessageService services.MessageServiceInterface + logger logger.LoggerInterface +} + +func NewSseSendMessageHandler( + service services.MessageServiceInterface, + logger logger.LoggerInterface, +) *SseSendMessageHandler { + return &SseSendMessageHandler{ + MessageService: service, + logger: logger, + } +} + +func (handler *SseSendMessageHandler) SseSendMessageController(w http.ResponseWriter, r *http.Request) { + userIdStr := mux.Vars(r)["user_id"] + sessionIdStr := mux.Vars(r)["session_id"] + + // Validate user_id + userId, err := uuid.Parse(userIdStr) + if userIdStr == "" || err != nil { + http.Error(w, "invalid user_id", http.StatusBadRequest) + return + } + + // Validate session_id + chatSessionID, err := uuid.Parse(sessionIdStr) + if sessionIdStr == "" || err != nil { + http.Error(w, "invalid session_id", http.StatusBadRequest) + return + } + + // Decode prompt from POST body + var req SendMessageRequest + if err := json.NewDecoder(r.Body).Decode(&req); err != nil { + http.Error(w, "invalid request body", http.StatusBadRequest) + return + } + + // Create the user's message + domainMessage := &domain.Message{ + ChatSessionID: chatSessionID, + Content: req.Content, + Sender: repositories.USER_SENDER, + } + + insertedUUID, err := handler.MessageService.CreateMessage(r.Context(), userId, domainMessage) + if err != nil { + handler.logger.Error("Error creating message", map[string]interface{}{"errorMessage": err.Error()}) + + switch e := err.(type) { + case customerrors.ResourceNotFoundErrorWrapper: + http.Error(w, "chat session not found: "+e.Error(), http.StatusNotFound) + case customerrors.UserMismatchError: + http.Error(w, "user mismatch: "+e.Error(), http.StatusForbidden) + default: + http.Error(w, "internal error: "+err.Error(), http.StatusInternalServerError) + } + return + } + + // Prepare SSE headers + w.Header().Set("Content-Type", "text/event-stream") + w.Header().Set("Cache-Control", "no-cache") + w.Header().Set("Connection", "keep-alive") + w.WriteHeader(http.StatusOK) + flusher, ok := w.(http.Flusher) + if !ok { + http.Error(w, "streaming unsupported", http.StatusInternalServerError) + return + } + + // Send initial user message + userMessageJson, _ := json.Marshal(MessageResponseFromModel(domainMessage)) + fmt.Fprintf(w, "data: {\"user_message\": %s}\n\n", userMessageJson) + flusher.Flush() + + // Stream system message token-by-token + tokenChan, errChan := handler.MessageService.StreamAnswerForMessage(r.Context(), insertedUUID) + + for { + select { + case token, ok := <-tokenChan: + if !ok { + fmt.Fprintf(w, "data: [DONE]\n\n") + flusher.Flush() + return + } + fmt.Fprintf(w, "data: %s\n\n", formatAsJsonToken(token)) + flusher.Flush() + + case err := <-errChan: + if err != nil { + handler.logger.Error("AI streaming error", map[string]interface{}{"errorMessage": err.Error()}) + errorJson, _ := json.Marshal(map[string]string{"error": err.Error()}) + fmt.Fprintf(w, "data: %s\n\n", errorJson) + flusher.Flush() + return + } + } + } +} + +func formatAsJsonToken(token string) string { + escaped := strings.ReplaceAll(token, `"`, `\"`) + return fmt.Sprintf("{\"token\": \"%s\"}", escaped) +} diff --git a/pkg/server/http/routes.go b/pkg/server/http/routes.go index b9c9968..dcf068f 100644 --- a/pkg/server/http/routes.go +++ b/pkg/server/http/routes.go @@ -70,6 +70,7 @@ func (s *Server) initializeRoutes() { getChatSessionHandler := chatSessionHandlers.NewGetChatSessionHandler(chatSessionService, s.logger) sendMessageHandler := chatSessionHandlers.NewSendMessageHandler(messageService, s.logger) wsSendMessageHandler := chatSessionHandlers.NewWsSendMessageHandler(messageService, s.logger, &upgrader) + sseSendMessageHandler := chatSessionHandlers.NewSseSendMessageHandler(messageService, s.logger) submitFeedbackHandler := chatSessionHandlers.NewSubmitFeedbackHandler(messageService, s.logger) protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionController).Methods("POST") @@ -82,4 +83,5 @@ func (s *Server) initializeRoutes() { protected.HandleFunc("/ws/add", wsAddHandler.AddController) protected.HandleFunc("/ws/subtract", wsSubtractHandler.SubtractController) protected.HandleFunc("/ws/users/{user_id}/chat-sessions/{session_id}/messages", wsSendMessageHandler.WsSendMessageController) + protected.HandleFunc("/sse/users/{user_id}/chat-sessions/{session_id}/messages", sseSendMessageHandler.SseSendMessageController) } From 37700cc8c05331bdc4d430484b5db62b7cd5af1d Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 13 Aug 2025 16:06:31 +0300 Subject: [PATCH 51/52] Adds dummy cors to BE --- go.mod | 2 ++ go.sum | 4 ++++ pkg/server/http/routes.go | 23 +++++++++++++++++++++++ 3 files changed, 29 insertions(+) diff --git a/go.mod b/go.mod index b493dbd..bdc6c6c 100644 --- a/go.mod +++ b/go.mod @@ -8,6 +8,7 @@ require ( github.com/DATA-DOG/go-sqlmock v1.5.2 github.com/golang-jwt/jwt/v4 v4.5.0 github.com/google/uuid v1.6.0 + github.com/gorilla/handlers v1.5.2 github.com/gorilla/mux v1.8.0 github.com/gorilla/websocket v1.5.3 github.com/joho/godotenv v1.5.1 @@ -32,6 +33,7 @@ require ( github.com/apapsch/go-jsonmerge/v2 v2.0.0 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/dlclark/regexp2 v1.10.0 // indirect + github.com/felixge/httpsnoop v1.0.3 // indirect github.com/go-openapi/jsonpointer v0.19.5 // indirect github.com/go-openapi/jsonreference v0.19.6 // indirect github.com/go-openapi/spec v0.20.4 // indirect diff --git a/go.sum b/go.sum index 1d594b8..a91395c 100644 --- a/go.sum +++ b/go.sum @@ -16,6 +16,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/dlclark/regexp2 v1.10.0 h1:+/GIL799phkJqYW+3YbOd8LCcbHzT0Pbo8zl70MHsq0= github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= +github.com/felixge/httpsnoop v1.0.3 h1:s/nj+GCswXYzN5v2DpNMuMQYe+0DDwt5WVCU6CWBdXk= +github.com/felixge/httpsnoop v1.0.3/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U= github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= @@ -34,6 +36,8 @@ github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/gorilla/handlers v1.5.2 h1:cLTUSsNkgcwhgRqvCNmdbRWG0A3N4F+M2nWKdScwyEE= +github.com/gorilla/handlers v1.5.2/go.mod h1:dX+xVpaxdSw+q0Qek8SSsl3dfMk3jNddUkMzo0GtH0w= github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg= diff --git a/pkg/server/http/routes.go b/pkg/server/http/routes.go index dcf068f..20b988c 100644 --- a/pkg/server/http/routes.go +++ b/pkg/server/http/routes.go @@ -1,6 +1,8 @@ package http import ( + gorillaHandlers "github.com/gorilla/handlers" + "github.com/gorilla/mux" "github.com/gorilla/websocket" "github.com/loukaspe/jedi-team-challenge/internal/core/services" httpHandlers "github.com/loukaspe/jedi-team-challenge/internal/handlers/http" @@ -57,6 +59,7 @@ func (s *Server) initializeRoutes() { jwtHandler := httpHandlers.NewJwtClaimsHandler(jwtService, s.logger) s.router.HandleFunc("/token", jwtHandler.JwtTokenController).Methods(http.MethodPost) + s.router.HandleFunc("/token", optionsHandlerForCors).Methods(http.MethodOptions) protected := s.router.PathPrefix("/").Subrouter() protected.Use(jwtMiddleware.AuthenticationMW) @@ -74,6 +77,7 @@ func (s *Server) initializeRoutes() { submitFeedbackHandler := chatSessionHandlers.NewSubmitFeedbackHandler(messageService, s.logger) protected.HandleFunc("/users/{user_id}/chat-sessions", createChatSessionHandler.CreateUserChatSessionController).Methods("POST") + protected.HandleFunc("/users/{user_id}/chat-sessions", optionsHandlerForCors).Methods(http.MethodOptions) protected.HandleFunc("/users/{user_id}/chat-sessions", getChatSessionHandler.GetUserChatSessionsController).Methods("GET") protected.HandleFunc("/users/{user_id}/chat-sessions/{session_id}/messages", sendMessageHandler.SendMessageController).Methods("POST") protected.HandleFunc("/users/{user_id}/chat-sessions/{session_id}/messages/{message_id}/feedback", submitFeedbackHandler.SubmitFeedbackController).Methods("POST") @@ -84,4 +88,23 @@ func (s *Server) initializeRoutes() { protected.HandleFunc("/ws/subtract", wsSubtractHandler.SubtractController) protected.HandleFunc("/ws/users/{user_id}/chat-sessions/{session_id}/messages", wsSendMessageHandler.WsSendMessageController) protected.HandleFunc("/sse/users/{user_id}/chat-sessions/{session_id}/messages", sseSendMessageHandler.SseSendMessageController) + protected.HandleFunc("/sse/users/{user_id}/chat-sessions/{session_id}/messages", optionsHandlerForCors).Methods(http.MethodOptions) + + s.router.Use(mux.CORSMethodMiddleware(s.router)) + + corsOrigins := gorillaHandlers.AllowedOrigins([]string{"http://localhost:3000"}) + corsMethods := gorillaHandlers.AllowedMethods([]string{"GET", "POST", "PUT", "DELETE", "OPTIONS"}) + corsHeaders := gorillaHandlers.AllowedHeaders([]string{"Content-Type", "Authorization"}) + corsExposedHeaders := gorillaHandlers.ExposedHeaders([]string{"Content-Type", "Authorization"}) + corsCredentials := gorillaHandlers.AllowCredentials() + corsHandler := gorillaHandlers.CORS(corsOrigins, corsMethods, corsHeaders, corsExposedHeaders, corsCredentials) + s.router.Use(corsHandler) +} + +func optionsHandlerForCors(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Access-Control-Allow-Origin", "http://localhost:3000") + w.Header().Set("Access-Control-Max-Age", "86400") + w.Header().Set("Access-Control-Allow-Credentials", "true") + w.Header().Set("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE, PATCH, OPTIONS") + w.Header().Set("Access-Control-Allow-Headers", "Content-Type, Authorization") } From 9565a085becbef09e4e80d6b5f00879062eb65a5 Mon Sep 17 00:00:00 2001 From: Loukas Peteinaris Date: Wed, 13 Aug 2025 16:07:07 +0300 Subject: [PATCH 52/52] Adds simple FE with SSE connection --- front-end/README.md | 70 + front-end/package-lock.json | 17855 +++++++++++++++++++++++ front-end/package.json | 47 + front-end/public/favicon.ico | Bin 0 -> 3870 bytes front-end/public/images/logo.svg | 1 + front-end/public/index.html | 43 + front-end/public/logo192.png | Bin 0 -> 5347 bytes front-end/public/logo512.png | Bin 0 -> 9664 bytes front-end/public/manifest.json | 25 + front-end/public/robots.txt | 3 + front-end/src/App.css | 5 + front-end/src/App.js | 216 + front-end/src/App.test.js | 8 + front-end/src/index.css | 13 + front-end/src/index.js | 20 + front-end/src/logo.svg | 1 + front-end/src/reportWebVitals.js | 13 + front-end/src/routes/ProtectedRoute.js | 10 + front-end/src/setupTests.js | 5 + front-end/src/styles/variables.css | 15 + 20 files changed, 18350 insertions(+) create mode 100644 front-end/README.md create mode 100644 front-end/package-lock.json create mode 100644 front-end/package.json create mode 100644 front-end/public/favicon.ico create mode 100644 front-end/public/images/logo.svg create mode 100644 front-end/public/index.html create mode 100644 front-end/public/logo192.png create mode 100644 front-end/public/logo512.png create mode 100644 front-end/public/manifest.json create mode 100644 front-end/public/robots.txt create mode 100644 front-end/src/App.css create mode 100644 front-end/src/App.js create mode 100644 front-end/src/App.test.js create mode 100644 front-end/src/index.css create mode 100644 front-end/src/index.js create mode 100644 front-end/src/logo.svg create mode 100644 front-end/src/reportWebVitals.js create mode 100644 front-end/src/routes/ProtectedRoute.js create mode 100644 front-end/src/setupTests.js create mode 100644 front-end/src/styles/variables.css diff --git a/front-end/README.md b/front-end/README.md new file mode 100644 index 0000000..58beeac --- /dev/null +++ b/front-end/README.md @@ -0,0 +1,70 @@ +# Getting Started with Create React App + +This project was bootstrapped with [Create React App](https://github.com/facebook/create-react-app). + +## Available Scripts + +In the project directory, you can run: + +### `npm start` + +Runs the app in the development mode.\ +Open [http://localhost:3000](http://localhost:3000) to view it in your browser. + +The page will reload when you make changes.\ +You may also see any lint errors in the console. + +### `npm test` + +Launches the test runner in the interactive watch mode.\ +See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information. + +### `npm run build` + +Builds the app for production to the `build` folder.\ +It correctly bundles React in production mode and optimizes the build for the best performance. + +The build is minified and the filenames include the hashes.\ +Your app is ready to be deployed! + +See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information. + +### `npm run eject` + +**Note: this is a one-way operation. Once you `eject`, you can't go back!** + +If you aren't satisfied with the build tool and configuration choices, you can `eject` at any time. This command will remove the single build dependency from your project. + +Instead, it will copy all the configuration files and the transitive dependencies (webpack, Babel, ESLint, etc) right into your project so you have full control over them. All of the commands except `eject` will still work, but they will point to the copied scripts so you can tweak them. At this point you're on your own. + +You don't have to ever use `eject`. The curated feature set is suitable for small and middle deployments, and you shouldn't feel obligated to use this feature. However we understand that this tool wouldn't be useful if you couldn't customize it when you are ready for it. + +## Learn More + +You can learn more in the [Create React App documentation](https://facebook.github.io/create-react-app/docs/getting-started). + +To learn React, check out the [React documentation](https://reactjs.org/). + +### Code Splitting + +This section has moved here: [https://facebook.github.io/create-react-app/docs/code-splitting](https://facebook.github.io/create-react-app/docs/code-splitting) + +### Analyzing the Bundle Size + +This section has moved here: [https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size](https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size) + +### Making a Progressive Web App + +This section has moved here: [https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app](https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app) + +### Advanced Configuration + +This section has moved here: [https://facebook.github.io/create-react-app/docs/advanced-configuration](https://facebook.github.io/create-react-app/docs/advanced-configuration) + +### Deployment + +This section has moved here: [https://facebook.github.io/create-react-app/docs/deployment](https://facebook.github.io/create-react-app/docs/deployment) + +### `npm run build` fails to minify + +This section has moved here: [https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify](https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify) diff --git a/front-end/package-lock.json b/front-end/package-lock.json new file mode 100644 index 0000000..7514987 --- /dev/null +++ b/front-end/package-lock.json @@ -0,0 +1,17855 @@ +{ + "name": "louk-chatwalker", + "version": "0.1.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "louk-chatwalker", + "version": "0.1.0", + "dependencies": { + "@fortawesome/free-solid-svg-icons": "^6.4.2", + "@fortawesome/react-fontawesome": "^0.2.0", + "@radix-ui/react-icons": "^1.3.2", + "@react-buddy/ide-toolbox": "^2.4.0", + "@testing-library/jest-dom": "^5.17.0", + "@testing-library/react": "^13.4.0", + "@testing-library/user-event": "^13.5.0", + "axios": "latest", + "jwt-decode": "latest", + "nanoid": "^5.0.1", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "react-router-dom": "^6.16.0", + "react-scripts": "5.0.1", + "universal-cookie": "^6.1.1", + "web-vitals": "^2.1.4" + } + }, + "node_modules/@aashutoshrathi/word-wrap": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", + "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@adobe/css-tools": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/@adobe/css-tools/-/css-tools-4.3.1.tgz", + "integrity": "sha512-/62yikz7NLScCGAAST5SHdnjaDJQBDq0M2muyRTpf2VQhw6StBg2ALiu73zSJQ4fMVLA+0uBhBHAle7Wg+2kSg==" + }, + "node_modules/@alloc/quick-lru": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz", + "integrity": "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@ampproject/remapping": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.2.1.tgz", + "integrity": "sha512-lFMjJTrFL3j7L9yBxwYfCq2k6qqwHyzuUl/XBnif78PWTJYyL/dfowQHWE3sp6U6ZzqWiiIZnpTMO96zhkjwtg==", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.0", + "@jridgewell/trace-mapping": "^0.3.9" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.22.10.tgz", + "integrity": "sha512-/KKIMG4UEL35WmI9OlvMhurwtytjvXoFcGNrOvyG9zIzA8YmPjVtIZUf7b05+TPO7G7/GEmLHDaoCgACHl9hhA==", + "dependencies": { + "@babel/highlight": "^7.22.10", + "chalk": "^2.4.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.22.9", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.22.9.tgz", + "integrity": "sha512-5UamI7xkUcJ3i9qVDS+KFDEK8/7oJ55/sJMB1Ge7IEapr7KfdfV/HErR+koZwOfd+SgtFKOKRhRakdg++DcJpQ==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.22.11.tgz", + "integrity": "sha512-lh7RJrtPdhibbxndr6/xx0w8+CVlY5FJZiaSz908Fpy+G0xkBFTvwLcKJFF4PJxVfGhVWNebikpWGnOoC71juQ==", + "dependencies": { + "@ampproject/remapping": "^2.2.0", + "@babel/code-frame": "^7.22.10", + "@babel/generator": "^7.22.10", + "@babel/helper-compilation-targets": "^7.22.10", + "@babel/helper-module-transforms": "^7.22.9", + "@babel/helpers": "^7.22.11", + "@babel/parser": "^7.22.11", + "@babel/template": "^7.22.5", + "@babel/traverse": "^7.22.11", + "@babel/types": "^7.22.11", + "convert-source-map": "^1.7.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/core/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/eslint-parser": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/eslint-parser/-/eslint-parser-7.22.11.tgz", + "integrity": "sha512-YjOYZ3j7TjV8OhLW6NCtyg8G04uStATEUe5eiLuCZaXz2VSDQ3dsAtm2D+TuQyAqNMUK2WacGo0/uma9Pein1w==", + "dependencies": { + "@nicolo-ribaudo/eslint-scope-5-internals": "5.1.1-v1", + "eslint-visitor-keys": "^2.1.0", + "semver": "^6.3.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || >=14.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.11.0", + "eslint": "^7.5.0 || ^8.0.0" + } + }, + "node_modules/@babel/eslint-parser/node_modules/eslint-visitor-keys": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", + "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", + "engines": { + "node": ">=10" + } + }, + "node_modules/@babel/eslint-parser/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/generator": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.22.10.tgz", + "integrity": "sha512-79KIf7YiWjjdZ81JnLujDRApWtl7BxTqWD88+FFdQEIOG8LJ0etDOM7CXuIgGJa55sGOwZVwuEsaLEm0PJ5/+A==", + "dependencies": { + "@babel/types": "^7.22.10", + "@jridgewell/gen-mapping": "^0.3.2", + "@jridgewell/trace-mapping": "^0.3.17", + "jsesc": "^2.5.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-annotate-as-pure": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.22.5.tgz", + "integrity": "sha512-LvBTxu8bQSQkcyKOU+a1btnNFQ1dMAd0R6PyW3arXes06F6QLWLIrd681bxRPIXlrMGR3XYnW9JyML7dP3qgxg==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-builder-binary-assignment-operator-visitor": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.22.10.tgz", + "integrity": "sha512-Av0qubwDQxC56DoUReVDeLfMEjYYSN1nZrTUrWkXd7hpU73ymRANkbuDm3yni9npkn+RXy9nNbEJZEzXr7xrfQ==", + "dependencies": { + "@babel/types": "^7.22.10" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.22.10.tgz", + "integrity": "sha512-JMSwHD4J7SLod0idLq5PKgI+6g/hLD/iuWBq08ZX49xE14VpVEojJ5rHWptpirV2j020MvypRLAXAO50igCJ5Q==", + "dependencies": { + "@babel/compat-data": "^7.22.9", + "@babel/helper-validator-option": "^7.22.5", + "browserslist": "^4.21.9", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-create-class-features-plugin": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.22.11.tgz", + "integrity": "sha512-y1grdYL4WzmUDBRGK0pDbIoFd7UZKoDurDzWEoNMYoj1EL+foGRQNyPWDcC+YyegN5y1DUsFFmzjGijB3nSVAQ==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "@babel/helper-environment-visitor": "^7.22.5", + "@babel/helper-function-name": "^7.22.5", + "@babel/helper-member-expression-to-functions": "^7.22.5", + "@babel/helper-optimise-call-expression": "^7.22.5", + "@babel/helper-replace-supers": "^7.22.9", + "@babel/helper-skip-transparent-expression-wrappers": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-create-class-features-plugin/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-create-regexp-features-plugin": { + "version": "7.22.9", + "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.22.9.tgz", + "integrity": "sha512-+svjVa/tFwsNSG4NEy1h85+HQ5imbT92Q5/bgtS7P0GTQlP8WuFdqsiABmQouhiFGyV66oGxZFpeYHza1rNsKw==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "regexpu-core": "^5.3.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-create-regexp-features-plugin/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-define-polyfill-provider": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/@babel/helper-define-polyfill-provider/-/helper-define-polyfill-provider-0.4.2.tgz", + "integrity": "sha512-k0qnnOqHn5dK9pZpfD5XXZ9SojAITdCKRn2Lp6rnDGzIbaP0rHyMPk/4wsSxVBVz4RfN0q6VpXWP2pDGIoQ7hw==", + "dependencies": { + "@babel/helper-compilation-targets": "^7.22.6", + "@babel/helper-plugin-utils": "^7.22.5", + "debug": "^4.1.1", + "lodash.debounce": "^4.0.8", + "resolve": "^1.14.2" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/@babel/helper-environment-visitor": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.5.tgz", + "integrity": "sha512-XGmhECfVA/5sAt+H+xpSg0mfrHq6FzNr9Oxh7PSEBBRUb/mL7Kz3NICXb194rCqAEdxkhPT1a88teizAFyvk8Q==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-function-name": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.22.5.tgz", + "integrity": "sha512-wtHSq6jMRE3uF2otvfuD3DIvVhOsSNshQl0Qrd7qC9oQJzHvOL4qQXlQn2916+CXGywIjpGuIkoyZRRxHPiNQQ==", + "dependencies": { + "@babel/template": "^7.22.5", + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-hoist-variables": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz", + "integrity": "sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-member-expression-to-functions": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.22.5.tgz", + "integrity": "sha512-aBiH1NKMG0H2cGZqspNvsaBe6wNGjbJjuLy29aU+eDZjSbbN53BaxlpB02xm9v34pLTZ1nIQPFYn2qMZoa5BQQ==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.22.5.tgz", + "integrity": "sha512-8Dl6+HD/cKifutF5qGd/8ZJi84QeAKh+CEe1sBzz8UayBBGg1dAIJrdHOcOM5b2MpzWL2yuotJTtGjETq0qjXg==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.22.9", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.22.9.tgz", + "integrity": "sha512-t+WA2Xn5K+rTeGtC8jCsdAH52bjggG5TKRuRrAGNM/mjIbO4GxvlLMFOEz9wXY5I2XQ60PMFsAG2WIcG82dQMQ==", + "dependencies": { + "@babel/helper-environment-visitor": "^7.22.5", + "@babel/helper-module-imports": "^7.22.5", + "@babel/helper-simple-access": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "@babel/helper-validator-identifier": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-optimise-call-expression": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.22.5.tgz", + "integrity": "sha512-HBwaojN0xFRx4yIvpwGqxiV2tUfl7401jlok564NgB9EHS1y6QT17FmKWm4ztqjeVdXLuC4fSvHc5ePpQjoTbw==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.22.5.tgz", + "integrity": "sha512-uLls06UVKgFG9QD4OeFYLEGteMIAa5kpTPcFL28yuCIIzsf6ZyKZMllKVOCZFhiZ5ptnwX4mtKdWCBE/uT4amg==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-remap-async-to-generator": { + "version": "7.22.9", + "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.22.9.tgz", + "integrity": "sha512-8WWC4oR4Px+tr+Fp0X3RHDVfINGpF3ad1HIbrc8A77epiR6eMMc6jsgozkzT2uDiOOdoS9cLIQ+XD2XvI2WSmQ==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "@babel/helper-environment-visitor": "^7.22.5", + "@babel/helper-wrap-function": "^7.22.9" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-replace-supers": { + "version": "7.22.9", + "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.22.9.tgz", + "integrity": "sha512-LJIKvvpgPOPUThdYqcX6IXRuIcTkcAub0IaDRGCZH0p5GPUp7PhRU9QVgFcDDd51BaPkk77ZjqFwh6DZTAEmGg==", + "dependencies": { + "@babel/helper-environment-visitor": "^7.22.5", + "@babel/helper-member-expression-to-functions": "^7.22.5", + "@babel/helper-optimise-call-expression": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-simple-access": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.22.5.tgz", + "integrity": "sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-skip-transparent-expression-wrappers": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.22.5.tgz", + "integrity": "sha512-tK14r66JZKiC43p8Ki33yLBVJKlQDFoA8GYN67lWCDCqoL6EMMSuM9b+Iff2jHaM/RRFYl7K+iiru7hbRqNx8Q==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-split-export-declaration": { + "version": "7.22.6", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.6.tgz", + "integrity": "sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==", + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.22.5.tgz", + "integrity": "sha512-mM4COjgZox8U+JcXQwPijIZLElkgEpO5rsERVDJTc2qfCDfERyob6k5WegS14SX18IIjv+XD+GrqNumY5JRCDw==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.5.tgz", + "integrity": "sha512-aJXu+6lErq8ltp+JhkJUfk1MTGyuA4v7f3pA+BJ5HLfNC6nAQ0Cpi9uOquUj8Hehg0aUiHzWQbOVJGao6ztBAQ==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.22.5.tgz", + "integrity": "sha512-R3oB6xlIVKUnxNUxbmgq7pKjxpru24zlimpE8WK47fACIlM0II/Hm1RS8IaOI7NgCr6LNS+jl5l75m20npAziw==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-wrap-function": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.22.10.tgz", + "integrity": "sha512-OnMhjWjuGYtdoO3FmsEFWvBStBAe2QOgwOLsLNDjN+aaiMD8InJk1/O3HSD8lkqTjCgg5YI34Tz15KNNA3p+nQ==", + "dependencies": { + "@babel/helper-function-name": "^7.22.5", + "@babel/template": "^7.22.5", + "@babel/types": "^7.22.10" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.22.11.tgz", + "integrity": "sha512-vyOXC8PBWaGc5h7GMsNx68OH33cypkEDJCHvYVVgVbbxJDROYVtexSk0gK5iCF1xNjRIN2s8ai7hwkWDq5szWg==", + "dependencies": { + "@babel/template": "^7.22.5", + "@babel/traverse": "^7.22.11", + "@babel/types": "^7.22.11" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/highlight": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.22.10.tgz", + "integrity": "sha512-78aUtVcT7MUscr0K5mIEnkwxPE0MaxkR5RxRwuHaQ+JuU5AmTPhY+do2mdzVTnIJJpyBglql2pehuBIWHug+WQ==", + "dependencies": { + "@babel/helper-validator-identifier": "^7.22.5", + "chalk": "^2.4.2", + "js-tokens": "^4.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.22.11.tgz", + "integrity": "sha512-R5zb8eJIBPJriQtbH/htEQy4k7E2dHWlD2Y2VT07JCzwYZHBxV5ZYtM0UhXSNMT74LyxuM+b1jdL7pSesXbC/g==", + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression/-/plugin-bugfix-safari-id-destructuring-collision-in-function-expression-7.22.5.tgz", + "integrity": "sha512-NP1M5Rf+u2Gw9qfSO4ihjcTGW5zXTi36ITLd4/EoAcEhIZ0yjMqmftDNl3QC19CX7olhrjpyU454g/2W7X0jvQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining/-/plugin-bugfix-v8-spread-parameters-in-optional-chaining-7.22.5.tgz", + "integrity": "sha512-31Bb65aZaUwqCbWMnZPduIZxCBngHFlzyN6Dq6KAJjtx+lx6ohKHubc61OomYi7XwVD4Ol0XCVz4h+pYFR048g==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-skip-transparent-expression-wrappers": "^7.22.5", + "@babel/plugin-transform-optional-chaining": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.13.0" + } + }, + "node_modules/@babel/plugin-proposal-class-properties": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.18.6.tgz", + "integrity": "sha512-cumfXOF0+nzZrrN8Rf0t7M+tF6sZc7vhQwYQck9q1/5w2OExlD+b4v4RpMJFaV1Z7WcDRgO6FqvxqxGlwo+RHQ==", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.18.6", + "@babel/helper-plugin-utils": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-decorators": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-decorators/-/plugin-proposal-decorators-7.22.10.tgz", + "integrity": "sha512-KxN6TqZzcFi4uD3UifqXElBTBNLAEH1l3vzMQj6JwJZbL2sZlThxSViOKCYY+4Ah4V4JhQ95IVB7s/Y6SJSlMQ==", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.22.10", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-replace-supers": "^7.22.9", + "@babel/helper-split-export-declaration": "^7.22.6", + "@babel/plugin-syntax-decorators": "^7.22.10" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-nullish-coalescing-operator": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.18.6.tgz", + "integrity": "sha512-wQxQzxYeJqHcfppzBDnm1yAY0jSRkUXR2z8RePZYrKwMKgMlE8+Z6LUno+bd6LvbGh8Gltvy74+9pIYkr+XkKA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.18.6", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-numeric-separator": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.18.6.tgz", + "integrity": "sha512-ozlZFogPqoLm8WBr5Z8UckIoE4YQ5KESVcNudyXOR8uqIkliTEgJ3RoketfG6pmzLdeZF0H/wjE9/cCEitBl7Q==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.18.6", + "@babel/plugin-syntax-numeric-separator": "^7.10.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-optional-chaining": { + "version": "7.21.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.21.0.tgz", + "integrity": "sha512-p4zeefM72gpmEe2fkUr/OnOXpWEf8nAgk7ZYVqqfFiyIG7oFfVZcCrU64hWn5xp4tQ9LkV4bTIa5rD0KANpKNA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.20.2", + "@babel/helper-skip-transparent-expression-wrappers": "^7.20.0", + "@babel/plugin-syntax-optional-chaining": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-private-methods": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.18.6.tgz", + "integrity": "sha512-nutsvktDItsNn4rpGItSNV2sz1XwS+nfU0Rg8aCx3W3NOKVzdMjJRu0O5OkgDp3ZGICSTbgRpxZoWsxoKRvbeA==", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.18.6", + "@babel/helper-plugin-utils": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-private-property-in-object": { + "version": "7.21.0-placeholder-for-preset-env.2", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.0-placeholder-for-preset-env.2.tgz", + "integrity": "sha512-SOSkfJDddaM7mak6cPEpswyTRnuRltl429hMraQEglW+OkovnCzsiszTmsrlY//qLFjCpQDFRvjdm2wA5pPm9w==", + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", + "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-static-block": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-static-block/-/plugin-syntax-class-static-block-7.14.5.tgz", + "integrity": "sha512-b+YyPmr6ldyNnM6sqYeMWE+bgJcJpO6yS4QD7ymxgH34GBPNDM/THBh8iunyvKIZztiwLH4CJZ0RxTk9emgpjw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-decorators": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-decorators/-/plugin-syntax-decorators-7.22.10.tgz", + "integrity": "sha512-z1KTVemBjnz+kSEilAsI4lbkPOl5TvJH7YDSY1CTIzvLWJ+KHXp+mRe8VPmfnyvqOPqar1V2gid2PleKzRUstQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-dynamic-import": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz", + "integrity": "sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-export-namespace-from": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz", + "integrity": "sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-flow": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-flow/-/plugin-syntax-flow-7.22.5.tgz", + "integrity": "sha512-9RdCl0i+q0QExayk2nOS7853w08yLucnnPML6EN9S8fgMPVtdLDCdx/cOQ/i44Lb9UeQX9A35yaqBBOMMZxPxQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-assertions": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-assertions/-/plugin-syntax-import-assertions-7.22.5.tgz", + "integrity": "sha512-rdV97N7KqsRzeNGoWUOK6yUsWarLjE5Su/Snk9IYPU9CwkWHs4t+rTGOvffTR8XGkJMTAdLfO0xVnXm8wugIJg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-attributes": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.22.5.tgz", + "integrity": "sha512-KwvoWDeNKPETmozyFE0P2rOLqh39EoQHNjqizrI5B8Vt0ZNS7M56s7dAiAqbYfiAYOuIzIh96z3iR2ktgu3tEg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", + "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.22.5.tgz", + "integrity": "sha512-gvyP4hZrgrs/wWMaocvxZ44Hw0b3W8Pe+cMxc8V1ULQ07oh8VNbIRaoD1LRZVTvD+0nieDKjfgKg89sD7rrKrg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", + "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", + "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-private-property-in-object": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-private-property-in-object/-/plugin-syntax-private-property-in-object-7.14.5.tgz", + "integrity": "sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.22.5.tgz", + "integrity": "sha512-1mS2o03i7t1c6VzH6fdQ3OA8tcEIxwG18zIPRp+UY1Ihv6W+XZzBCVxExF9upussPXJ0xE9XRHwMoNs1ep/nRQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-unicode-sets-regex": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-unicode-sets-regex/-/plugin-syntax-unicode-sets-regex-7.18.6.tgz", + "integrity": "sha512-727YkEAPwSIQTv5im8QHz3upqp92JTWhidIC81Tdx4VJYIte/VndKf1qKrfnnhPLiPghStWfvC/iFaMCQu7Nqg==", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.18.6", + "@babel/helper-plugin-utils": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-arrow-functions": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.22.5.tgz", + "integrity": "sha512-26lTNXoVRdAnsaDXPpvCNUq+OVWEVC6bx7Vvz9rC53F2bagUWW4u4ii2+h8Fejfh7RYqPxn+libeFBBck9muEw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-async-generator-functions": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-generator-functions/-/plugin-transform-async-generator-functions-7.22.11.tgz", + "integrity": "sha512-0pAlmeRJn6wU84zzZsEOx1JV1Jf8fqO9ok7wofIJwUnplYo247dcd24P+cMJht7ts9xkzdtB0EPHmOb7F+KzXw==", + "dependencies": { + "@babel/helper-environment-visitor": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-remap-async-to-generator": "^7.22.9", + "@babel/plugin-syntax-async-generators": "^7.8.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-async-to-generator": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.22.5.tgz", + "integrity": "sha512-b1A8D8ZzE/VhNDoV1MSJTnpKkCG5bJo+19R4o4oy03zM7ws8yEMK755j61Dc3EyvdysbqH5BOOTquJ7ZX9C6vQ==", + "dependencies": { + "@babel/helper-module-imports": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-remap-async-to-generator": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoped-functions": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.22.5.tgz", + "integrity": "sha512-tdXZ2UdknEKQWKJP1KMNmuF5Lx3MymtMN/pvA+p/VEkhK8jVcQ1fzSy8KM9qRYhAf2/lV33hoMPKI/xaI9sADA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoping": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.22.10.tgz", + "integrity": "sha512-1+kVpGAOOI1Albt6Vse7c8pHzcZQdQKW+wJH+g8mCaszOdDVwRXa/slHPqIw+oJAJANTKDMuM2cBdV0Dg618Vg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-class-properties": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-class-properties/-/plugin-transform-class-properties-7.22.5.tgz", + "integrity": "sha512-nDkQ0NfkOhPTq8YCLiWNxp1+f9fCobEjCb0n8WdbNUBc4IB5V7P1QnX9IjpSoquKrXF5SKojHleVNs2vGeHCHQ==", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-class-static-block": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-class-static-block/-/plugin-transform-class-static-block-7.22.11.tgz", + "integrity": "sha512-GMM8gGmqI7guS/llMFk1bJDkKfn3v3C4KHK9Yg1ey5qcHcOlKb0QvcMrgzvxo+T03/4szNh5lghY+fEC98Kq9g==", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.22.11", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-class-static-block": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.12.0" + } + }, + "node_modules/@babel/plugin-transform-classes": { + "version": "7.22.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.22.6.tgz", + "integrity": "sha512-58EgM6nuPNG6Py4Z3zSuu0xWu2VfodiMi72Jt5Kj2FECmaYk1RrTXA45z6KBFsu9tRgwQDwIiY4FXTt+YsSFAQ==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "@babel/helper-compilation-targets": "^7.22.6", + "@babel/helper-environment-visitor": "^7.22.5", + "@babel/helper-function-name": "^7.22.5", + "@babel/helper-optimise-call-expression": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-replace-supers": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-computed-properties": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.22.5.tgz", + "integrity": "sha512-4GHWBgRf0krxPX+AaPtgBAlTgTeZmqDynokHOX7aqqAB4tHs3U2Y02zH6ETFdLZGcg9UQSD1WCmkVrE9ErHeOg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/template": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-destructuring": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.22.10.tgz", + "integrity": "sha512-dPJrL0VOyxqLM9sritNbMSGx/teueHF/htMKrPT7DNxccXxRDPYqlgPFFdr8u+F+qUZOkZoXue/6rL5O5GduEw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-dotall-regex": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.22.5.tgz", + "integrity": "sha512-5/Yk9QxCQCl+sOIB1WelKnVRxTJDSAIxtJLL2/pqL14ZVlbH0fUQUZa/T5/UnQtBNgghR7mfB8ERBKyKPCi7Vw==", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-duplicate-keys": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.22.5.tgz", + "integrity": "sha512-dEnYD+9BBgld5VBXHnF/DbYGp3fqGMsyxKbtD1mDyIA7AkTSpKXFhCVuj/oQVOoALfBs77DudA0BE4d5mcpmqw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-dynamic-import": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dynamic-import/-/plugin-transform-dynamic-import-7.22.11.tgz", + "integrity": "sha512-g/21plo58sfteWjaO0ZNVb+uEOkJNjAaHhbejrnBmu011l/eNDScmkbjCC3l4FKb10ViaGU4aOkFznSu2zRHgA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-dynamic-import": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-exponentiation-operator": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.22.5.tgz", + "integrity": "sha512-vIpJFNM/FjZ4rh1myqIya9jXwrwwgFRHPjT3DkUA9ZLHuzox8jiXkOLvwm1H+PQIP3CqfC++WPKeuDi0Sjdj1g==", + "dependencies": { + "@babel/helper-builder-binary-assignment-operator-visitor": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-export-namespace-from": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-export-namespace-from/-/plugin-transform-export-namespace-from-7.22.11.tgz", + "integrity": "sha512-xa7aad7q7OiT8oNZ1mU7NrISjlSkVdMbNxn9IuLZyL9AJEhs1Apba3I+u5riX1dIkdptP5EKDG5XDPByWxtehw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-export-namespace-from": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-flow-strip-types": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-flow-strip-types/-/plugin-transform-flow-strip-types-7.22.5.tgz", + "integrity": "sha512-tujNbZdxdG0/54g/oua8ISToaXTFBf8EnSb5PgQSciIXWOWKX3S4+JR7ZE9ol8FZwf9kxitzkGQ+QWeov/mCiA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-flow": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-for-of": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.22.5.tgz", + "integrity": "sha512-3kxQjX1dU9uudwSshyLeEipvrLjBCVthCgeTp6CzE/9JYrlAIaeekVxRpCWsDDfYTfRZRoCeZatCQvwo+wvK8A==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-function-name": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.22.5.tgz", + "integrity": "sha512-UIzQNMS0p0HHiQm3oelztj+ECwFnj+ZRV4KnguvlsD2of1whUeM6o7wGNj6oLwcDoAXQ8gEqfgC24D+VdIcevg==", + "dependencies": { + "@babel/helper-compilation-targets": "^7.22.5", + "@babel/helper-function-name": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-json-strings": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-json-strings/-/plugin-transform-json-strings-7.22.11.tgz", + "integrity": "sha512-CxT5tCqpA9/jXFlme9xIBCc5RPtdDq3JpkkhgHQqtDdiTnTI0jtZ0QzXhr5DILeYifDPp2wvY2ad+7+hLMW5Pw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-json-strings": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-literals": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.22.5.tgz", + "integrity": "sha512-fTLj4D79M+mepcw3dgFBTIDYpbcB9Sm0bpm4ppXPaO+U+PKFFyV9MGRvS0gvGw62sd10kT5lRMKXAADb9pWy8g==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-logical-assignment-operators": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-logical-assignment-operators/-/plugin-transform-logical-assignment-operators-7.22.11.tgz", + "integrity": "sha512-qQwRTP4+6xFCDV5k7gZBF3C31K34ut0tbEcTKxlX/0KXxm9GLcO14p570aWxFvVzx6QAfPgq7gaeIHXJC8LswQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-member-expression-literals": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.22.5.tgz", + "integrity": "sha512-RZEdkNtzzYCFl9SE9ATaUMTj2hqMb4StarOJLrZRbqqU4HSBE7UlBw9WBWQiDzrJZJdUWiMTVDI6Gv/8DPvfew==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-amd": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.22.5.tgz", + "integrity": "sha512-R+PTfLTcYEmb1+kK7FNkhQ1gP4KgjpSO6HfH9+f8/yfp2Nt3ggBjiVpRwmwTlfqZLafYKJACy36yDXlEmI9HjQ==", + "dependencies": { + "@babel/helper-module-transforms": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-commonjs": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.22.11.tgz", + "integrity": "sha512-o2+bg7GDS60cJMgz9jWqRUsWkMzLCxp+jFDeDUT5sjRlAxcJWZ2ylNdI7QQ2+CH5hWu7OnN+Cv3htt7AkSf96g==", + "dependencies": { + "@babel/helper-module-transforms": "^7.22.9", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-simple-access": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-systemjs": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.22.11.tgz", + "integrity": "sha512-rIqHmHoMEOhI3VkVf5jQ15l539KrwhzqcBO6wdCNWPWc/JWt9ILNYNUssbRpeq0qWns8svuw8LnMNCvWBIJ8wA==", + "dependencies": { + "@babel/helper-hoist-variables": "^7.22.5", + "@babel/helper-module-transforms": "^7.22.9", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-validator-identifier": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-umd": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.22.5.tgz", + "integrity": "sha512-+S6kzefN/E1vkSsKx8kmQuqeQsvCKCd1fraCM7zXm4SFoggI099Tr4G8U81+5gtMdUeMQ4ipdQffbKLX0/7dBQ==", + "dependencies": { + "@babel/helper-module-transforms": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-named-capturing-groups-regex": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.22.5.tgz", + "integrity": "sha512-YgLLKmS3aUBhHaxp5hi1WJTgOUb/NCuDHzGT9z9WTt3YG+CPRhJs6nprbStx6DnWM4dh6gt7SU3sZodbZ08adQ==", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-new-target": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.22.5.tgz", + "integrity": "sha512-AsF7K0Fx/cNKVyk3a+DW0JLo+Ua598/NxMRvxDnkpCIGFh43+h/v2xyhRUYf6oD8gE4QtL83C7zZVghMjHd+iw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-nullish-coalescing-operator": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-nullish-coalescing-operator/-/plugin-transform-nullish-coalescing-operator-7.22.11.tgz", + "integrity": "sha512-YZWOw4HxXrotb5xsjMJUDlLgcDXSfO9eCmdl1bgW4+/lAGdkjaEvOnQ4p5WKKdUgSzO39dgPl0pTnfxm0OAXcg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-numeric-separator": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-numeric-separator/-/plugin-transform-numeric-separator-7.22.11.tgz", + "integrity": "sha512-3dzU4QGPsILdJbASKhF/V2TVP+gJya1PsueQCxIPCEcerqF21oEcrob4mzjsp2Py/1nLfF5m+xYNMDpmA8vffg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-numeric-separator": "^7.10.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-object-rest-spread": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-rest-spread/-/plugin-transform-object-rest-spread-7.22.11.tgz", + "integrity": "sha512-nX8cPFa6+UmbepISvlf5jhQyaC7ASs/7UxHmMkuJ/k5xSHvDPPaibMo+v3TXwU/Pjqhep/nFNpd3zn4YR59pnw==", + "dependencies": { + "@babel/compat-data": "^7.22.9", + "@babel/helper-compilation-targets": "^7.22.10", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-transform-parameters": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-object-super": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.22.5.tgz", + "integrity": "sha512-klXqyaT9trSjIUrcsYIfETAzmOEZL3cBYqOYLJxBHfMFFggmXOv+NYSX/Jbs9mzMVESw/WycLFPRx8ba/b2Ipw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-replace-supers": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-optional-catch-binding": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-optional-catch-binding/-/plugin-transform-optional-catch-binding-7.22.11.tgz", + "integrity": "sha512-rli0WxesXUeCJnMYhzAglEjLWVDF6ahb45HuprcmQuLidBJFWjNnOzssk2kuc6e33FlLaiZhG/kUIzUMWdBKaQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-optional-chaining": { + "version": "7.22.12", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-optional-chaining/-/plugin-transform-optional-chaining-7.22.12.tgz", + "integrity": "sha512-7XXCVqZtyFWqjDsYDY4T45w4mlx1rf7aOgkc/Ww76xkgBiOlmjPkx36PBLHa1k1rwWvVgYMPsbuVnIamx2ZQJw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-skip-transparent-expression-wrappers": "^7.22.5", + "@babel/plugin-syntax-optional-chaining": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-parameters": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.22.5.tgz", + "integrity": "sha512-AVkFUBurORBREOmHRKo06FjHYgjrabpdqRSwq6+C7R5iTCZOsM4QbcB27St0a4U6fffyAOqh3s/qEfybAhfivg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-private-methods": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-private-methods/-/plugin-transform-private-methods-7.22.5.tgz", + "integrity": "sha512-PPjh4gyrQnGe97JTalgRGMuU4icsZFnWkzicB/fUtzlKUqvsWBKEpPPfr5a2JiyirZkHxnAqkQMO5Z5B2kK3fA==", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-private-property-in-object": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-private-property-in-object/-/plugin-transform-private-property-in-object-7.22.11.tgz", + "integrity": "sha512-sSCbqZDBKHetvjSwpyWzhuHkmW5RummxJBVbYLkGkaiTOWGxml7SXt0iWa03bzxFIx7wOj3g/ILRd0RcJKBeSQ==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "@babel/helper-create-class-features-plugin": "^7.22.11", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-property-literals": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.22.5.tgz", + "integrity": "sha512-TiOArgddK3mK/x1Qwf5hay2pxI6wCZnvQqrFSqbtg1GLl2JcNMitVH/YnqjP+M31pLUeTfzY1HAXFDnUBV30rQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-constant-elements": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-constant-elements/-/plugin-transform-react-constant-elements-7.22.5.tgz", + "integrity": "sha512-BF5SXoO+nX3h5OhlN78XbbDrBOffv+AxPP2ENaJOVqjWCgBDeOY3WcaUcddutGSfoap+5NEQ/q/4I3WZIvgkXA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-display-name": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-display-name/-/plugin-transform-react-display-name-7.22.5.tgz", + "integrity": "sha512-PVk3WPYudRF5z4GKMEYUrLjPl38fJSKNaEOkFuoprioowGuWN6w2RKznuFNSlJx7pzzXXStPUnNSOEO0jL5EVw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx/-/plugin-transform-react-jsx-7.22.5.tgz", + "integrity": "sha512-rog5gZaVbUip5iWDMTYbVM15XQq+RkUKhET/IHR6oizR+JEoN6CAfTTuHcK4vwUyzca30qqHqEpzBOnaRMWYMA==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "@babel/helper-module-imports": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-jsx": "^7.22.5", + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-development": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-development/-/plugin-transform-react-jsx-development-7.22.5.tgz", + "integrity": "sha512-bDhuzwWMuInwCYeDeMzyi7TaBgRQei6DqxhbyniL7/VG4RSS7HtSL2QbY4eESy1KJqlWt8g3xeEBGPuo+XqC8A==", + "dependencies": { + "@babel/plugin-transform-react-jsx": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-pure-annotations": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-pure-annotations/-/plugin-transform-react-pure-annotations-7.22.5.tgz", + "integrity": "sha512-gP4k85wx09q+brArVinTXhWiyzLl9UpmGva0+mWyKxk6JZequ05x3eUcIUE+FyttPKJFRRVtAvQaJ6YF9h1ZpA==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-regenerator": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.22.10.tgz", + "integrity": "sha512-F28b1mDt8KcT5bUyJc/U9nwzw6cV+UmTeRlXYIl2TNqMMJif0Jeey9/RQ3C4NOd2zp0/TRsDns9ttj2L523rsw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "regenerator-transform": "^0.15.2" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-reserved-words": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.22.5.tgz", + "integrity": "sha512-DTtGKFRQUDm8svigJzZHzb/2xatPc6TzNvAIJ5GqOKDsGFYgAskjRulbR/vGsPKq3OPqtexnz327qYpP57RFyA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-runtime": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.22.10.tgz", + "integrity": "sha512-RchI7HePu1eu0CYNKHHHQdfenZcM4nz8rew5B1VWqeRKdcwW5aQ5HeG9eTUbWiAS1UrmHVLmoxTWHt3iLD/NhA==", + "dependencies": { + "@babel/helper-module-imports": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5", + "babel-plugin-polyfill-corejs2": "^0.4.5", + "babel-plugin-polyfill-corejs3": "^0.8.3", + "babel-plugin-polyfill-regenerator": "^0.5.2", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-runtime/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/plugin-transform-shorthand-properties": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.22.5.tgz", + "integrity": "sha512-vM4fq9IXHscXVKzDv5itkO1X52SmdFBFcMIBZ2FRn2nqVYqw6dBexUgMvAjHW+KXpPPViD/Yo3GrDEBaRC0QYA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-spread": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.22.5.tgz", + "integrity": "sha512-5ZzDQIGyvN4w8+dMmpohL6MBo+l2G7tfC/O2Dg7/hjpgeWvUx8FzfeOKxGog9IimPa4YekaQ9PlDqTLOljkcxg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-skip-transparent-expression-wrappers": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-sticky-regex": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.22.5.tgz", + "integrity": "sha512-zf7LuNpHG0iEeiyCNwX4j3gDg1jgt1k3ZdXBKbZSoA3BbGQGvMiSvfbZRR3Dr3aeJe3ooWFZxOOG3IRStYp2Bw==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-template-literals": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.22.5.tgz", + "integrity": "sha512-5ciOehRNf+EyUeewo8NkbQiUs4d6ZxiHo6BcBcnFlgiJfu16q0bQUw9Jvo0b0gBKFG1SMhDSjeKXSYuJLeFSMA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-typeof-symbol": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.22.5.tgz", + "integrity": "sha512-bYkI5lMzL4kPii4HHEEChkD0rkc+nvnlR6+o/qdqR6zrm0Sv/nodmyLhlq2DO0YKLUNd2VePmPRjJXSBh9OIdA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-typescript": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.22.11.tgz", + "integrity": "sha512-0E4/L+7gfvHub7wsbTv03oRtD69X31LByy44fGmFzbZScpupFByMcgCJ0VbBTkzyjSJKuRoGN8tcijOWKTmqOA==", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.22.5", + "@babel/helper-create-class-features-plugin": "^7.22.11", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/plugin-syntax-typescript": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-escapes": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.22.10.tgz", + "integrity": "sha512-lRfaRKGZCBqDlRU3UIFovdp9c9mEvlylmpod0/OatICsSfuQ9YFthRo1tpTkGsklEefZdqlEFdY4A2dwTb6ohg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-property-regex": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-property-regex/-/plugin-transform-unicode-property-regex-7.22.5.tgz", + "integrity": "sha512-HCCIb+CbJIAE6sXn5CjFQXMwkCClcOfPCzTlilJ8cUatfzwHlWQkbtV0zD338u9dZskwvuOYTuuaMaA8J5EI5A==", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-regex": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.22.5.tgz", + "integrity": "sha512-028laaOKptN5vHJf9/Arr/HiJekMd41hOEZYvNsrsXqJ7YPYuX2bQxh31fkZzGmq3YqHRJzYFFAVYvKfMPKqyg==", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-sets-regex": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-sets-regex/-/plugin-transform-unicode-sets-regex-7.22.5.tgz", + "integrity": "sha512-lhMfi4FC15j13eKrh3DnYHjpGj6UKQHtNKTbtc1igvAhRy4+kLhV07OpLcsN0VgDEw/MjAvJO4BdMJsHwMhzCg==", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.22.5", + "@babel/helper-plugin-utils": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/preset-env": { + "version": "7.22.10", + "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.22.10.tgz", + "integrity": "sha512-riHpLb1drNkpLlocmSyEg4oYJIQFeXAK/d7rI6mbD0XsvoTOOweXDmQPG/ErxsEhWk3rl3Q/3F6RFQlVFS8m0A==", + "dependencies": { + "@babel/compat-data": "^7.22.9", + "@babel/helper-compilation-targets": "^7.22.10", + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-validator-option": "^7.22.5", + "@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression": "^7.22.5", + "@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining": "^7.22.5", + "@babel/plugin-proposal-private-property-in-object": "7.21.0-placeholder-for-preset-env.2", + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-class-properties": "^7.12.13", + "@babel/plugin-syntax-class-static-block": "^7.14.5", + "@babel/plugin-syntax-dynamic-import": "^7.8.3", + "@babel/plugin-syntax-export-namespace-from": "^7.8.3", + "@babel/plugin-syntax-import-assertions": "^7.22.5", + "@babel/plugin-syntax-import-attributes": "^7.22.5", + "@babel/plugin-syntax-import-meta": "^7.10.4", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5", + "@babel/plugin-syntax-top-level-await": "^7.14.5", + "@babel/plugin-syntax-unicode-sets-regex": "^7.18.6", + "@babel/plugin-transform-arrow-functions": "^7.22.5", + "@babel/plugin-transform-async-generator-functions": "^7.22.10", + "@babel/plugin-transform-async-to-generator": "^7.22.5", + "@babel/plugin-transform-block-scoped-functions": "^7.22.5", + "@babel/plugin-transform-block-scoping": "^7.22.10", + "@babel/plugin-transform-class-properties": "^7.22.5", + "@babel/plugin-transform-class-static-block": "^7.22.5", + "@babel/plugin-transform-classes": "^7.22.6", + "@babel/plugin-transform-computed-properties": "^7.22.5", + "@babel/plugin-transform-destructuring": "^7.22.10", + "@babel/plugin-transform-dotall-regex": "^7.22.5", + "@babel/plugin-transform-duplicate-keys": "^7.22.5", + "@babel/plugin-transform-dynamic-import": "^7.22.5", + "@babel/plugin-transform-exponentiation-operator": "^7.22.5", + "@babel/plugin-transform-export-namespace-from": "^7.22.5", + "@babel/plugin-transform-for-of": "^7.22.5", + "@babel/plugin-transform-function-name": "^7.22.5", + "@babel/plugin-transform-json-strings": "^7.22.5", + "@babel/plugin-transform-literals": "^7.22.5", + "@babel/plugin-transform-logical-assignment-operators": "^7.22.5", + "@babel/plugin-transform-member-expression-literals": "^7.22.5", + "@babel/plugin-transform-modules-amd": "^7.22.5", + "@babel/plugin-transform-modules-commonjs": "^7.22.5", + "@babel/plugin-transform-modules-systemjs": "^7.22.5", + "@babel/plugin-transform-modules-umd": "^7.22.5", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.22.5", + "@babel/plugin-transform-new-target": "^7.22.5", + "@babel/plugin-transform-nullish-coalescing-operator": "^7.22.5", + "@babel/plugin-transform-numeric-separator": "^7.22.5", + "@babel/plugin-transform-object-rest-spread": "^7.22.5", + "@babel/plugin-transform-object-super": "^7.22.5", + "@babel/plugin-transform-optional-catch-binding": "^7.22.5", + "@babel/plugin-transform-optional-chaining": "^7.22.10", + "@babel/plugin-transform-parameters": "^7.22.5", + "@babel/plugin-transform-private-methods": "^7.22.5", + "@babel/plugin-transform-private-property-in-object": "^7.22.5", + "@babel/plugin-transform-property-literals": "^7.22.5", + "@babel/plugin-transform-regenerator": "^7.22.10", + "@babel/plugin-transform-reserved-words": "^7.22.5", + "@babel/plugin-transform-shorthand-properties": "^7.22.5", + "@babel/plugin-transform-spread": "^7.22.5", + "@babel/plugin-transform-sticky-regex": "^7.22.5", + "@babel/plugin-transform-template-literals": "^7.22.5", + "@babel/plugin-transform-typeof-symbol": "^7.22.5", + "@babel/plugin-transform-unicode-escapes": "^7.22.10", + "@babel/plugin-transform-unicode-property-regex": "^7.22.5", + "@babel/plugin-transform-unicode-regex": "^7.22.5", + "@babel/plugin-transform-unicode-sets-regex": "^7.22.5", + "@babel/preset-modules": "0.1.6-no-external-plugins", + "@babel/types": "^7.22.10", + "babel-plugin-polyfill-corejs2": "^0.4.5", + "babel-plugin-polyfill-corejs3": "^0.8.3", + "babel-plugin-polyfill-regenerator": "^0.5.2", + "core-js-compat": "^3.31.0", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/preset-env/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/preset-modules": { + "version": "0.1.6-no-external-plugins", + "resolved": "https://registry.npmjs.org/@babel/preset-modules/-/preset-modules-0.1.6-no-external-plugins.tgz", + "integrity": "sha512-HrcgcIESLm9aIR842yhJ5RWan/gebQUJ6E/E5+rf0y9o6oj7w0Br+sWuL6kEQ/o/AdfvR1Je9jG18/gnpwjEyA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/types": "^7.4.4", + "esutils": "^2.0.2" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/@babel/preset-react": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/preset-react/-/preset-react-7.22.5.tgz", + "integrity": "sha512-M+Is3WikOpEJHgR385HbuCITPTaPRaNkibTEa9oiofmJvIsrceb4yp9RL9Kb+TE8LznmeyZqpP+Lopwcx59xPQ==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-validator-option": "^7.22.5", + "@babel/plugin-transform-react-display-name": "^7.22.5", + "@babel/plugin-transform-react-jsx": "^7.22.5", + "@babel/plugin-transform-react-jsx-development": "^7.22.5", + "@babel/plugin-transform-react-pure-annotations": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/preset-typescript": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/preset-typescript/-/preset-typescript-7.22.11.tgz", + "integrity": "sha512-tWY5wyCZYBGY7IlalfKI1rLiGlIfnwsRHZqlky0HVv8qviwQ1Uo/05M6+s+TcTCVa6Bmoo2uJW5TMFX6Wa4qVg==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.22.5", + "@babel/helper-validator-option": "^7.22.5", + "@babel/plugin-syntax-jsx": "^7.22.5", + "@babel/plugin-transform-modules-commonjs": "^7.22.11", + "@babel/plugin-transform-typescript": "^7.22.11" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/regjsgen": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/@babel/regjsgen/-/regjsgen-0.8.0.tgz", + "integrity": "sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA==" + }, + "node_modules/@babel/runtime": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.22.11.tgz", + "integrity": "sha512-ee7jVNlWN09+KftVOu9n7S8gQzD/Z6hN/I8VBRXW4P1+Xe7kJGXMwu8vds4aGIMHZnNbdpSWCfZZtinytpcAvA==", + "dependencies": { + "regenerator-runtime": "^0.14.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/template": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.22.5.tgz", + "integrity": "sha512-X7yV7eiwAxdj9k94NEylvbVHLiVG1nvzCV2EAowhxLTwODV1jl9UzZ48leOC0sH7OnuHrIkllaBgneUykIcZaw==", + "dependencies": { + "@babel/code-frame": "^7.22.5", + "@babel/parser": "^7.22.5", + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.22.11.tgz", + "integrity": "sha512-mzAenteTfomcB7mfPtyi+4oe5BZ6MXxWcn4CX+h4IRJ+OOGXBrWU6jDQavkQI9Vuc5P+donFabBfFCcmWka9lQ==", + "dependencies": { + "@babel/code-frame": "^7.22.10", + "@babel/generator": "^7.22.10", + "@babel/helper-environment-visitor": "^7.22.5", + "@babel/helper-function-name": "^7.22.5", + "@babel/helper-hoist-variables": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "@babel/parser": "^7.22.11", + "@babel/types": "^7.22.11", + "debug": "^4.1.0", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.22.11", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.22.11.tgz", + "integrity": "sha512-siazHiGuZRz9aB9NpHy9GOs9xiQPKnMzgdr493iI1M67vRXpnEq8ZOOKzezC5q7zwuQ6sDhdSp4SD9ixKSqKZg==", + "dependencies": { + "@babel/helper-string-parser": "^7.22.5", + "@babel/helper-validator-identifier": "^7.22.5", + "to-fast-properties": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@bcoe/v8-coverage": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", + "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==" + }, + "node_modules/@csstools/normalize.css": { + "version": "12.0.0", + "resolved": "https://registry.npmjs.org/@csstools/normalize.css/-/normalize.css-12.0.0.tgz", + "integrity": "sha512-M0qqxAcwCsIVfpFQSlGN5XjXWu8l5JDZN+fPt1LeW5SZexQTgnaEvgXAY+CeygRw0EeppWHi12JxESWiWrB0Sg==" + }, + "node_modules/@csstools/postcss-cascade-layers": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@csstools/postcss-cascade-layers/-/postcss-cascade-layers-1.1.1.tgz", + "integrity": "sha512-+KdYrpKC5TgomQr2DlZF4lDEpHcoxnj5IGddYYfBWJAKfj1JtuHUIqMa+E1pJJ+z3kvDViWMqyqPlG4Ja7amQA==", + "dependencies": { + "@csstools/selector-specificity": "^2.0.2", + "postcss-selector-parser": "^6.0.10" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-color-function": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@csstools/postcss-color-function/-/postcss-color-function-1.1.1.tgz", + "integrity": "sha512-Bc0f62WmHdtRDjf5f3e2STwRAl89N2CLb+9iAwzrv4L2hncrbDwnQD9PCq0gtAt7pOI2leIV08HIBUd4jxD8cw==", + "dependencies": { + "@csstools/postcss-progressive-custom-properties": "^1.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-font-format-keywords": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@csstools/postcss-font-format-keywords/-/postcss-font-format-keywords-1.0.1.tgz", + "integrity": "sha512-ZgrlzuUAjXIOc2JueK0X5sZDjCtgimVp/O5CEqTcs5ShWBa6smhWYbS0x5cVc/+rycTDbjjzoP0KTDnUneZGOg==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-hwb-function": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@csstools/postcss-hwb-function/-/postcss-hwb-function-1.0.2.tgz", + "integrity": "sha512-YHdEru4o3Rsbjmu6vHy4UKOXZD+Rn2zmkAmLRfPet6+Jz4Ojw8cbWxe1n42VaXQhD3CQUXXTooIy8OkVbUcL+w==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-ic-unit": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@csstools/postcss-ic-unit/-/postcss-ic-unit-1.0.1.tgz", + "integrity": "sha512-Ot1rcwRAaRHNKC9tAqoqNZhjdYBzKk1POgWfhN4uCOE47ebGcLRqXjKkApVDpjifL6u2/55ekkpnFcp+s/OZUw==", + "dependencies": { + "@csstools/postcss-progressive-custom-properties": "^1.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-is-pseudo-class": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/@csstools/postcss-is-pseudo-class/-/postcss-is-pseudo-class-2.0.7.tgz", + "integrity": "sha512-7JPeVVZHd+jxYdULl87lvjgvWldYu+Bc62s9vD/ED6/QTGjy0jy0US/f6BG53sVMTBJ1lzKZFpYmofBN9eaRiA==", + "dependencies": { + "@csstools/selector-specificity": "^2.0.0", + "postcss-selector-parser": "^6.0.10" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-nested-calc": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@csstools/postcss-nested-calc/-/postcss-nested-calc-1.0.0.tgz", + "integrity": "sha512-JCsQsw1wjYwv1bJmgjKSoZNvf7R6+wuHDAbi5f/7MbFhl2d/+v+TvBTU4BJH3G1X1H87dHl0mh6TfYogbT/dJQ==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-normalize-display-values": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@csstools/postcss-normalize-display-values/-/postcss-normalize-display-values-1.0.1.tgz", + "integrity": "sha512-jcOanIbv55OFKQ3sYeFD/T0Ti7AMXc9nM1hZWu8m/2722gOTxFg7xYu4RDLJLeZmPUVQlGzo4jhzvTUq3x4ZUw==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-oklab-function": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@csstools/postcss-oklab-function/-/postcss-oklab-function-1.1.1.tgz", + "integrity": "sha512-nJpJgsdA3dA9y5pgyb/UfEzE7W5Ka7u0CX0/HIMVBNWzWemdcTH3XwANECU6anWv/ao4vVNLTMxhiPNZsTK6iA==", + "dependencies": { + "@csstools/postcss-progressive-custom-properties": "^1.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-progressive-custom-properties": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@csstools/postcss-progressive-custom-properties/-/postcss-progressive-custom-properties-1.3.0.tgz", + "integrity": "sha512-ASA9W1aIy5ygskZYuWams4BzafD12ULvSypmaLJT2jvQ8G0M3I8PRQhC0h7mG0Z3LI05+agZjqSR9+K9yaQQjA==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.3" + } + }, + "node_modules/@csstools/postcss-stepped-value-functions": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@csstools/postcss-stepped-value-functions/-/postcss-stepped-value-functions-1.0.1.tgz", + "integrity": "sha512-dz0LNoo3ijpTOQqEJLY8nyaapl6umbmDcgj4AD0lgVQ572b2eqA1iGZYTTWhrcrHztWDDRAX2DGYyw2VBjvCvQ==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-text-decoration-shorthand": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@csstools/postcss-text-decoration-shorthand/-/postcss-text-decoration-shorthand-1.0.0.tgz", + "integrity": "sha512-c1XwKJ2eMIWrzQenN0XbcfzckOLLJiczqy+YvfGmzoVXd7pT9FfObiSEfzs84bpE/VqfpEuAZ9tCRbZkZxxbdw==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-trigonometric-functions": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@csstools/postcss-trigonometric-functions/-/postcss-trigonometric-functions-1.0.2.tgz", + "integrity": "sha512-woKaLO///4bb+zZC2s80l+7cm07M7268MsyG3M0ActXXEFi6SuhvriQYcb58iiKGbjwwIU7n45iRLEHypB47Og==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/postcss-unset-value": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@csstools/postcss-unset-value/-/postcss-unset-value-1.0.2.tgz", + "integrity": "sha512-c8J4roPBILnelAsdLr4XOAR/GsTm0GJi4XpcfvoWk3U6KiTCqiFYc63KhRMQQX35jYMp4Ao8Ij9+IZRgMfJp1g==", + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/@csstools/selector-specificity": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@csstools/selector-specificity/-/selector-specificity-2.2.0.tgz", + "integrity": "sha512-+OJ9konv95ClSTOJCmMZqpd5+YGsB2S+x6w3E1oaM8UuR5j8nTNHYSz8c9BEPGDOCMQYIEEGlVPj/VY64iTbGw==", + "engines": { + "node": "^14 || ^16 || >=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss-selector-parser": "^6.0.10" + } + }, + "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==", + "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/regexpp": { + "version": "4.8.0", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.8.0.tgz", + "integrity": "sha512-JylOEEzDiOryeUnFbQz+oViCXS0KsvR1mvHkoMiu5+UiBvy+RYX7tzlIIIEstF/gVa2tj9AQXk3dgnxv6KxhFg==", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.2.tgz", + "integrity": "sha512-+wvgpDsrB1YqAMdEUCcnTlpfVBH7Vqn6A/NT3D8WVXFIaKMlErPIZT3oCIAVCOtarRpMtelZLqJeU3t7WY6X6g==", + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.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": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/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==" + }, + "node_modules/@eslint/eslintrc/node_modules/globals": { + "version": "13.21.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.21.0.tgz", + "integrity": "sha512-ybyme3s4yy/t/3s35bewwXKOf7cvzfreG2lH0lZl0JB7I4GxRP2ghxOK/Nb9EkRXdbBXZLfq/p/0W2JUONB/Gg==", + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@eslint/eslintrc/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==", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/@eslint/eslintrc/node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@eslint/js": { + "version": "8.48.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.48.0.tgz", + "integrity": "sha512-ZSjtmelB7IJfWD2Fvb7+Z+ChTIKWq6kjda95fLcQKNS5aheVHn4IkfgRQE3sIIzTcSLwLcLZUD9UBt+V7+h+Pw==", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + } + }, + "node_modules/@fortawesome/fontawesome-common-types": { + "version": "6.4.2", + "resolved": "https://registry.npmjs.org/@fortawesome/fontawesome-common-types/-/fontawesome-common-types-6.4.2.tgz", + "integrity": "sha512-1DgP7f+XQIJbLFCTX1V2QnxVmpLdKdzzo2k8EmvDOePfchaIGQ9eCHj2up3/jNEbZuBqel5OxiaOJf37TWauRA==", + "hasInstallScript": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/@fortawesome/fontawesome-svg-core": { + "version": "6.4.2", + "resolved": "https://registry.npmjs.org/@fortawesome/fontawesome-svg-core/-/fontawesome-svg-core-6.4.2.tgz", + "integrity": "sha512-gjYDSKv3TrM2sLTOKBc5rH9ckje8Wrwgx1CxAPbN5N3Fm4prfi7NsJVWd1jklp7i5uSCVwhZS5qlhMXqLrpAIg==", + "hasInstallScript": true, + "peer": true, + "dependencies": { + "@fortawesome/fontawesome-common-types": "6.4.2" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@fortawesome/free-solid-svg-icons": { + "version": "6.4.2", + "resolved": "https://registry.npmjs.org/@fortawesome/free-solid-svg-icons/-/free-solid-svg-icons-6.4.2.tgz", + "integrity": "sha512-sYwXurXUEQS32fZz9hVCUUv/xu49PEJEyUOsA51l6PU/qVgfbTb2glsTEaJngVVT8VqBATRIdh7XVgV1JF1LkA==", + "hasInstallScript": true, + "dependencies": { + "@fortawesome/fontawesome-common-types": "6.4.2" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@fortawesome/react-fontawesome": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/@fortawesome/react-fontawesome/-/react-fontawesome-0.2.0.tgz", + "integrity": "sha512-uHg75Rb/XORTtVt7OS9WoK8uM276Ufi7gCzshVWkUJbHhh3svsUUeqXerrM96Wm7fRiDzfKRwSoahhMIkGAYHw==", + "dependencies": { + "prop-types": "^15.8.1" + }, + "peerDependencies": { + "@fortawesome/fontawesome-svg-core": "~1 || ~6", + "react": ">=16.3" + } + }, + "node_modules/@humanwhocodes/config-array": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.10.tgz", + "integrity": "sha512-KVVjQmNUepDVGXNuoRRdmmEjruj0KfiGSbS8LVc12LMsWDQzRXJ0qdhN8L8uUigKpfEHRhlaQFY0ib1tnUbNeQ==", + "dependencies": { + "@humanwhocodes/object-schema": "^1.2.1", + "debug": "^4.1.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.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==", + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/object-schema": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz", + "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==" + }, + "node_modules/@istanbuljs/load-nyc-config": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", + "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", + "dependencies": { + "camelcase": "^5.3.1", + "find-up": "^4.1.0", + "get-package-type": "^0.1.0", + "js-yaml": "^3.13.1", + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "engines": { + "node": ">=6" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/schema": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", + "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/console": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-27.5.1.tgz", + "integrity": "sha512-kZ/tNpS3NXn0mlXXXPNuDZnb4c0oZ20r4K5eemM2k30ZC3G0T02nXUvyhf5YdbXWHPEJLc9qGLxEZ216MdL+Zg==", + "dependencies": { + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "jest-message-util": "^27.5.1", + "jest-util": "^27.5.1", + "slash": "^3.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/console/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/console/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/@jest/console/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@jest/console/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==" + }, + "node_modules/@jest/console/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/console/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/core": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-27.5.1.tgz", + "integrity": "sha512-AK6/UTrvQD0Cd24NSqmIA6rKsu0tKIxfiCducZvqxYdmMisOYAsdItspT+fQDQYARPf8XgjAFZi0ogW2agH5nQ==", + "dependencies": { + "@jest/console": "^27.5.1", + "@jest/reporters": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "emittery": "^0.8.1", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-changed-files": "^27.5.1", + "jest-config": "^27.5.1", + "jest-haste-map": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-resolve-dependencies": "^27.5.1", + "jest-runner": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "jest-watcher": "^27.5.1", + "micromatch": "^4.0.4", + "rimraf": "^3.0.0", + "slash": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/core/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/core/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/@jest/core/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@jest/core/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==" + }, + "node_modules/@jest/core/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/core/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/environment": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-27.5.1.tgz", + "integrity": "sha512-/WQjhPJe3/ghaol/4Bq480JKXV/Rfw8nQdN7f41fM8VDHLcxKXou6QyXAh3EFr9/bVG3x74z1NWDkP87EiY8gA==", + "dependencies": { + "@jest/fake-timers": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "jest-mock": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/expect-utils": { + "version": "29.6.4", + "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-29.6.4.tgz", + "integrity": "sha512-FEhkJhqtvBwgSpiTrocquJCdXPsyvNKcl/n7A3u7X4pVoF4bswm11c9d4AV+kfq2Gpv/mM8x7E7DsRvH+djkrg==", + "dependencies": { + "jest-get-type": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect-utils/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/fake-timers": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-27.5.1.tgz", + "integrity": "sha512-/aPowoolwa07k7/oM3aASneNeBGCmGQsc3ugN4u6s4C/+s5M64MFo/+djTdiwcbQlRfFElGuDXWzaWj6QgKObQ==", + "dependencies": { + "@jest/types": "^27.5.1", + "@sinonjs/fake-timers": "^8.0.1", + "@types/node": "*", + "jest-message-util": "^27.5.1", + "jest-mock": "^27.5.1", + "jest-util": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/globals": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-27.5.1.tgz", + "integrity": "sha512-ZEJNB41OBQQgGzgyInAv0UUfDDj3upmHydjieSxFvTRuZElrx7tXg/uVQ5hYVEwiXs3+aMsAeEc9X7xiSKCm4Q==", + "dependencies": { + "@jest/environment": "^27.5.1", + "@jest/types": "^27.5.1", + "expect": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/reporters": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-27.5.1.tgz", + "integrity": "sha512-cPXh9hWIlVJMQkVk84aIvXuBB4uQQmFqZiacloFuGiP3ah1sbCxCosidXFDfqG8+6fO1oR2dTJTlsOy4VFmUfw==", + "dependencies": { + "@bcoe/v8-coverage": "^0.2.3", + "@jest/console": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "collect-v8-coverage": "^1.0.0", + "exit": "^0.1.2", + "glob": "^7.1.2", + "graceful-fs": "^4.2.9", + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-instrument": "^5.1.0", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^4.0.0", + "istanbul-reports": "^3.1.3", + "jest-haste-map": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", + "slash": "^3.0.0", + "source-map": "^0.6.0", + "string-length": "^4.0.1", + "terminal-link": "^2.0.0", + "v8-to-istanbul": "^8.1.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/reporters/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/reporters/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/@jest/reporters/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@jest/reporters/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==" + }, + "node_modules/@jest/reporters/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/reporters/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@jest/reporters/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/schemas": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-28.1.3.tgz", + "integrity": "sha512-/l/VWsdt/aBXgjshLWOFyFt3IVdYypu5y2Wn2rOO1un6nkqIn8SLXzgIMYXFyYsRWDyF5EthmKJMIdJvk08grg==", + "dependencies": { + "@sinclair/typebox": "^0.24.1" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/@jest/source-map": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-27.5.1.tgz", + "integrity": "sha512-y9NIHUYF3PJRlHk98NdC/N1gl88BL08aQQgu4k4ZopQkCw9t9cV8mtl3TV8b/YCB8XaVTFrmUTAJvjsntDireg==", + "dependencies": { + "callsites": "^3.0.0", + "graceful-fs": "^4.2.9", + "source-map": "^0.6.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/source-map/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@jest/test-result": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-27.5.1.tgz", + "integrity": "sha512-EW35l2RYFUcUQxFJz5Cv5MTOxlJIQs4I7gxzi2zVU7PJhOwfYq1MdC5nhSmYjX1gmMmLPvB3sIaC+BkcHRBfag==", + "dependencies": { + "@jest/console": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/istanbul-lib-coverage": "^2.0.0", + "collect-v8-coverage": "^1.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/test-sequencer": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-27.5.1.tgz", + "integrity": "sha512-LCheJF7WB2+9JuCS7VB/EmGIdQuhtqjRNI9A43idHv3E4KltCTsPsLxvdaubFHSYwY/fNjMWjl6vNRhDiN7vpQ==", + "dependencies": { + "@jest/test-result": "^27.5.1", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-runtime": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/transform": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-27.5.1.tgz", + "integrity": "sha512-ipON6WtYgl/1329g5AIJVbUuEh0wZVbdpGwC99Jw4LwuoBNS95MVphU6zOeD9pDkon+LLbFL7lOQRapbB8SCHw==", + "dependencies": { + "@babel/core": "^7.1.0", + "@jest/types": "^27.5.1", + "babel-plugin-istanbul": "^6.1.1", + "chalk": "^4.0.0", + "convert-source-map": "^1.4.0", + "fast-json-stable-stringify": "^2.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-util": "^27.5.1", + "micromatch": "^4.0.4", + "pirates": "^4.0.4", + "slash": "^3.0.0", + "source-map": "^0.6.1", + "write-file-atomic": "^3.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/transform/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/transform/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/@jest/transform/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@jest/transform/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==" + }, + "node_modules/@jest/transform/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/transform/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@jest/transform/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/types": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.5.1.tgz", + "integrity": "sha512-Cx46iJ9QpwQTjIdq5VJu2QTMMs3QlEjI0x1QbBP5W1+nMzyc2XmimiRR/CbX9TO0cPTeUlxWMOu8mslYsJ8DEw==", + "dependencies": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^16.0.0", + "chalk": "^4.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@jest/types/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/types/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/@jest/types/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@jest/types/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==" + }, + "node_modules/@jest/types/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/types/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.3.tgz", + "integrity": "sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ==", + "dependencies": { + "@jridgewell/set-array": "^1.0.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.9" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz", + "integrity": "sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/set-array": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.1.2.tgz", + "integrity": "sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw==", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/source-map": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.5.tgz", + "integrity": "sha512-UTYAUj/wviwdsMfzoSJspJxbkH5o1snzwX0//0ENX1u/55kkZZkcTZP6u9bwKGkv+dkk9at4m1Cpt0uY80kcpQ==", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.0", + "@jridgewell/trace-mapping": "^0.3.9" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.4.15", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", + "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.19", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.19.tgz", + "integrity": "sha512-kf37QtfW+Hwx/buWGMPcR60iF9ziHa6r/CZJIHbmcm4+0qrXiVdxegAH0F6yddEVQ7zdkjcGCgCzUu+BcbhQxw==", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@leichtgewicht/ip-codec": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/@leichtgewicht/ip-codec/-/ip-codec-2.0.4.tgz", + "integrity": "sha512-Hcv+nVC0kZnQ3tD9GVu5xSMR4VVYOteQIr/hwFPVEvPdlXqgGEuRjiheChHgdM+JyqdgNcmzZOX/tnl0JOiI7A==" + }, + "node_modules/@nicolo-ribaudo/eslint-scope-5-internals": { + "version": "5.1.1-v1", + "resolved": "https://registry.npmjs.org/@nicolo-ribaudo/eslint-scope-5-internals/-/eslint-scope-5-internals-5.1.1-v1.tgz", + "integrity": "sha512-54/JRvkLIzzDWshCWfuhadfrfZVPiElY8Fcgmg1HroEly/EDSszzhBAsarCux+D/kOslTRquNzuyGSmUSTTHGg==", + "dependencies": { + "eslint-scope": "5.1.1" + } + }, + "node_modules/@nicolo-ribaudo/eslint-scope-5-internals/node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/@nicolo-ribaudo/eslint-scope-5-internals/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "engines": { + "node": ">=4.0" + } + }, + "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==", + "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==", + "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==", + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@pmmmwh/react-refresh-webpack-plugin": { + "version": "0.5.11", + "resolved": "https://registry.npmjs.org/@pmmmwh/react-refresh-webpack-plugin/-/react-refresh-webpack-plugin-0.5.11.tgz", + "integrity": "sha512-7j/6vdTym0+qZ6u4XbSAxrWBGYSdCfTzySkj7WAFgDLmSyWlOrWvpyzxlFh5jtw9dn0oL/jtW+06XfFiisN3JQ==", + "dependencies": { + "ansi-html-community": "^0.0.8", + "common-path-prefix": "^3.0.0", + "core-js-pure": "^3.23.3", + "error-stack-parser": "^2.0.6", + "find-up": "^5.0.0", + "html-entities": "^2.1.0", + "loader-utils": "^2.0.4", + "schema-utils": "^3.0.0", + "source-map": "^0.7.3" + }, + "engines": { + "node": ">= 10.13" + }, + "peerDependencies": { + "@types/webpack": "4.x || 5.x", + "react-refresh": ">=0.10.0 <1.0.0", + "sockjs-client": "^1.4.0", + "type-fest": ">=0.17.0 <5.0.0", + "webpack": ">=4.43.0 <6.0.0", + "webpack-dev-server": "3.x || 4.x", + "webpack-hot-middleware": "2.x", + "webpack-plugin-serve": "0.x || 1.x" + }, + "peerDependenciesMeta": { + "@types/webpack": { + "optional": true + }, + "sockjs-client": { + "optional": true + }, + "type-fest": { + "optional": true + }, + "webpack-dev-server": { + "optional": true + }, + "webpack-hot-middleware": { + "optional": true + }, + "webpack-plugin-serve": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-icons": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-icons/-/react-icons-1.3.2.tgz", + "integrity": "sha512-fyQIhGDhzfc9pK2kH6Pl9c4BDJGfMkPqkyIgYDthyNYoNg3wVhoJMMh19WS4Up/1KMPFVpNsT2q3WmXn2N1m6g==", + "license": "MIT", + "peerDependencies": { + "react": "^16.x || ^17.x || ^18.x || ^19.0.0 || ^19.0.0-rc" + } + }, + "node_modules/@react-buddy/ide-toolbox": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/@react-buddy/ide-toolbox/-/ide-toolbox-2.4.0.tgz", + "integrity": "sha512-TWHX6gwa0Gop7215uHhjFMbYLLdjM/b9rr0wYE3E0m7GNJ56gbPpbZiq86w9uI8zksl827acqGeT437MkuO64w==", + "peerDependencies": { + "react": "^17.0.0 || ^18.0.0" + } + }, + "node_modules/@remix-run/router": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.9.0.tgz", + "integrity": "sha512-bV63itrKBC0zdT27qYm6SDZHlkXwFL1xMBuhkn+X7l0+IIhNaH5wuuvZKp6eKhCD4KFhujhfhCT1YxXW6esUIA==", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@rollup/plugin-babel": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/@rollup/plugin-babel/-/plugin-babel-5.3.1.tgz", + "integrity": "sha512-WFfdLWU/xVWKeRQnKmIAQULUI7Il0gZnBIH/ZFO069wYIfPu+8zrfp/KMW0atmELoRDq8FbiP3VCss9MhCut7Q==", + "dependencies": { + "@babel/helper-module-imports": "^7.10.4", + "@rollup/pluginutils": "^3.1.0" + }, + "engines": { + "node": ">= 10.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0", + "@types/babel__core": "^7.1.9", + "rollup": "^1.20.0||^2.0.0" + }, + "peerDependenciesMeta": { + "@types/babel__core": { + "optional": true + } + } + }, + "node_modules/@rollup/plugin-node-resolve": { + "version": "11.2.1", + "resolved": "https://registry.npmjs.org/@rollup/plugin-node-resolve/-/plugin-node-resolve-11.2.1.tgz", + "integrity": "sha512-yc2n43jcqVyGE2sqV5/YCmocy9ArjVAP/BeXyTtADTBBX6V0e5UMqwO8CdQ0kzjb6zu5P1qMzsScCMRvE9OlVg==", + "dependencies": { + "@rollup/pluginutils": "^3.1.0", + "@types/resolve": "1.17.1", + "builtin-modules": "^3.1.0", + "deepmerge": "^4.2.2", + "is-module": "^1.0.0", + "resolve": "^1.19.0" + }, + "engines": { + "node": ">= 10.0.0" + }, + "peerDependencies": { + "rollup": "^1.20.0||^2.0.0" + } + }, + "node_modules/@rollup/plugin-replace": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/@rollup/plugin-replace/-/plugin-replace-2.4.2.tgz", + "integrity": "sha512-IGcu+cydlUMZ5En85jxHH4qj2hta/11BHq95iHEyb2sbgiN0eCdzvUcHw5gt9pBL5lTi4JDYJ1acCoMGpTvEZg==", + "dependencies": { + "@rollup/pluginutils": "^3.1.0", + "magic-string": "^0.25.7" + }, + "peerDependencies": { + "rollup": "^1.20.0 || ^2.0.0" + } + }, + "node_modules/@rollup/pluginutils": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-3.1.0.tgz", + "integrity": "sha512-GksZ6pr6TpIjHm8h9lSQ8pi8BE9VeubNT0OMJ3B5uZJ8pz73NPiqOtCog/x2/QzM1ENChPKxMDhiQuRHsqc+lg==", + "dependencies": { + "@types/estree": "0.0.39", + "estree-walker": "^1.0.1", + "picomatch": "^2.2.2" + }, + "engines": { + "node": ">= 8.0.0" + }, + "peerDependencies": { + "rollup": "^1.20.0||^2.0.0" + } + }, + "node_modules/@rollup/pluginutils/node_modules/@types/estree": { + "version": "0.0.39", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-0.0.39.tgz", + "integrity": "sha512-EYNwp3bU+98cpU4lAWYYL7Zz+2gryWH1qbdDTidVd6hkiR6weksdbMadyXKXNPEkQFhXM+hVO9ZygomHXp+AIw==" + }, + "node_modules/@rushstack/eslint-patch": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/@rushstack/eslint-patch/-/eslint-patch-1.3.3.tgz", + "integrity": "sha512-0xd7qez0AQ+MbHatZTlI1gu5vkG8r7MYRUJAHPAHJBmGLs16zpkrpAVLvjQKQOqaXPDUBwOiJzNc00znHSCVBw==" + }, + "node_modules/@sinclair/typebox": { + "version": "0.24.51", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.24.51.tgz", + "integrity": "sha512-1P1OROm/rdubP5aFDSZQILU0vrLCJ4fvHt6EoqHEM+2D/G5MK3bIaymUKLit8Js9gbns5UyJnkP/TZROLw4tUA==" + }, + "node_modules/@sinonjs/commons": { + "version": "1.8.6", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.6.tgz", + "integrity": "sha512-Ky+XkAkqPZSm3NLBeUng77EBQl3cmeJhITaGHdYH8kjVB+aun3S4XBRti2zt17mtt0mIUDiNxYeoJm6drVvBJQ==", + "dependencies": { + "type-detect": "4.0.8" + } + }, + "node_modules/@sinonjs/fake-timers": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-8.1.0.tgz", + "integrity": "sha512-OAPJUAtgeINhh/TAlUID4QTs53Njm7xzddaVlEs/SXwgtiD1tW22zAB/W1wdqfrpmikgaWQ9Fw6Ws+hsiRm5Vg==", + "dependencies": { + "@sinonjs/commons": "^1.7.0" + } + }, + "node_modules/@surma/rollup-plugin-off-main-thread": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/@surma/rollup-plugin-off-main-thread/-/rollup-plugin-off-main-thread-2.2.3.tgz", + "integrity": "sha512-lR8q/9W7hZpMWweNiAKU7NQerBnzQQLvi8qnTDU/fxItPhtZVMbPV3lbCwjhIlNBe9Bbr5V+KHshvWmVSG9cxQ==", + "dependencies": { + "ejs": "^3.1.6", + "json5": "^2.2.0", + "magic-string": "^0.25.0", + "string.prototype.matchall": "^4.0.6" + } + }, + "node_modules/@svgr/babel-plugin-add-jsx-attribute": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-add-jsx-attribute/-/babel-plugin-add-jsx-attribute-5.4.0.tgz", + "integrity": "sha512-ZFf2gs/8/6B8PnSofI0inYXr2SDNTDScPXhN7k5EqD4aZ3gi6u+rbmZHVB8IM3wDyx8ntKACZbtXSm7oZGRqVg==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-plugin-remove-jsx-attribute": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-remove-jsx-attribute/-/babel-plugin-remove-jsx-attribute-5.4.0.tgz", + "integrity": "sha512-yaS4o2PgUtwLFGTKbsiAy6D0o3ugcUhWK0Z45umJ66EPWunAz9fuFw2gJuje6wqQvQWOTJvIahUwndOXb7QCPg==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-plugin-remove-jsx-empty-expression": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-remove-jsx-empty-expression/-/babel-plugin-remove-jsx-empty-expression-5.0.1.tgz", + "integrity": "sha512-LA72+88A11ND/yFIMzyuLRSMJ+tRKeYKeQ+mR3DcAZ5I4h5CPWN9AHyUzJbWSYp/u2u0xhmgOe0+E41+GjEueA==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-plugin-replace-jsx-attribute-value": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-replace-jsx-attribute-value/-/babel-plugin-replace-jsx-attribute-value-5.0.1.tgz", + "integrity": "sha512-PoiE6ZD2Eiy5mK+fjHqwGOS+IXX0wq/YDtNyIgOrc6ejFnxN4b13pRpiIPbtPwHEc+NT2KCjteAcq33/F1Y9KQ==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-plugin-svg-dynamic-title": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-svg-dynamic-title/-/babel-plugin-svg-dynamic-title-5.4.0.tgz", + "integrity": "sha512-zSOZH8PdZOpuG1ZVx/cLVePB2ibo3WPpqo7gFIjLV9a0QsuQAzJiwwqmuEdTaW2pegyBE17Uu15mOgOcgabQZg==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-plugin-svg-em-dimensions": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-svg-em-dimensions/-/babel-plugin-svg-em-dimensions-5.4.0.tgz", + "integrity": "sha512-cPzDbDA5oT/sPXDCUYoVXEmm3VIoAWAPT6mSPTJNbQaBNUuEKVKyGH93oDY4e42PYHRW67N5alJx/eEol20abw==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-plugin-transform-react-native-svg": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-transform-react-native-svg/-/babel-plugin-transform-react-native-svg-5.4.0.tgz", + "integrity": "sha512-3eYP/SaopZ41GHwXma7Rmxcv9uRslRDTY1estspeB1w1ueZWd/tPlMfEOoccYpEMZU3jD4OU7YitnXcF5hLW2Q==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-plugin-transform-svg-component": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/@svgr/babel-plugin-transform-svg-component/-/babel-plugin-transform-svg-component-5.5.0.tgz", + "integrity": "sha512-q4jSH1UUvbrsOtlo/tKcgSeiCHRSBdXoIoqX1pgcKK/aU3JD27wmMKwGtpB8qRYUYoyXvfGxUVKchLuR5pB3rQ==", + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/babel-preset": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/@svgr/babel-preset/-/babel-preset-5.5.0.tgz", + "integrity": "sha512-4FiXBjvQ+z2j7yASeGPEi8VD/5rrGQk4Xrq3EdJmoZgz/tpqChpo5hgXDvmEauwtvOc52q8ghhZK4Oy7qph4ig==", + "dependencies": { + "@svgr/babel-plugin-add-jsx-attribute": "^5.4.0", + "@svgr/babel-plugin-remove-jsx-attribute": "^5.4.0", + "@svgr/babel-plugin-remove-jsx-empty-expression": "^5.0.1", + "@svgr/babel-plugin-replace-jsx-attribute-value": "^5.0.1", + "@svgr/babel-plugin-svg-dynamic-title": "^5.4.0", + "@svgr/babel-plugin-svg-em-dimensions": "^5.4.0", + "@svgr/babel-plugin-transform-react-native-svg": "^5.4.0", + "@svgr/babel-plugin-transform-svg-component": "^5.5.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/core": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/@svgr/core/-/core-5.5.0.tgz", + "integrity": "sha512-q52VOcsJPvV3jO1wkPtzTuKlvX7Y3xIcWRpCMtBF3MrteZJtBfQw/+u0B1BHy5ColpQc1/YVTrPEtSYIMNZlrQ==", + "dependencies": { + "@svgr/plugin-jsx": "^5.5.0", + "camelcase": "^6.2.0", + "cosmiconfig": "^7.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/hast-util-to-babel-ast": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/@svgr/hast-util-to-babel-ast/-/hast-util-to-babel-ast-5.5.0.tgz", + "integrity": "sha512-cAaR/CAiZRB8GP32N+1jocovUtvlj0+e65TB50/6Lcime+EA49m/8l+P2ko+XPJ4dw3xaPS3jOL4F2X4KWxoeQ==", + "dependencies": { + "@babel/types": "^7.12.6" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/plugin-jsx": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/@svgr/plugin-jsx/-/plugin-jsx-5.5.0.tgz", + "integrity": "sha512-V/wVh33j12hGh05IDg8GpIUXbjAPnTdPTKuP4VNLggnwaHMPNQNae2pRnyTAILWCQdz5GyMqtO488g7CKM8CBA==", + "dependencies": { + "@babel/core": "^7.12.3", + "@svgr/babel-preset": "^5.5.0", + "@svgr/hast-util-to-babel-ast": "^5.5.0", + "svg-parser": "^2.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/plugin-svgo": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/@svgr/plugin-svgo/-/plugin-svgo-5.5.0.tgz", + "integrity": "sha512-r5swKk46GuQl4RrVejVwpeeJaydoxkdwkM1mBKOgJLBUJPGaLci6ylg/IjhrRsREKDkr4kbMWdgOtbXEh0fyLQ==", + "dependencies": { + "cosmiconfig": "^7.0.0", + "deepmerge": "^4.2.2", + "svgo": "^1.2.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@svgr/webpack": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/@svgr/webpack/-/webpack-5.5.0.tgz", + "integrity": "sha512-DOBOK255wfQxguUta2INKkzPj6AIS6iafZYiYmHn6W3pHlycSRRlvWKCfLDG10fXfLWqE3DJHgRUOyJYmARa7g==", + "dependencies": { + "@babel/core": "^7.12.3", + "@babel/plugin-transform-react-constant-elements": "^7.12.1", + "@babel/preset-env": "^7.12.1", + "@babel/preset-react": "^7.12.5", + "@svgr/core": "^5.5.0", + "@svgr/plugin-jsx": "^5.5.0", + "@svgr/plugin-svgo": "^5.5.0", + "loader-utils": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/gregberge" + } + }, + "node_modules/@testing-library/dom": { + "version": "9.3.1", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-9.3.1.tgz", + "integrity": "sha512-0DGPd9AR3+iDTjGoMpxIkAsUihHZ3Ai6CneU6bRRrffXMgzCdlNk43jTrD2/5LT6CBb3MWTP8v510JzYtahD2w==", + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.10.4", + "@babel/runtime": "^7.12.5", + "@types/aria-query": "^5.0.1", + "aria-query": "5.1.3", + "chalk": "^4.1.0", + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "pretty-format": "^27.0.2" + }, + "engines": { + "node": ">=14" + } + }, + "node_modules/@testing-library/dom/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==", + "peer": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@testing-library/dom/node_modules/aria-query": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.1.3.tgz", + "integrity": "sha512-R5iJ5lkuHybztUfuOAznmboyjWq8O6sqNqtK7CLOqdydi54VNbORp49mb14KbWgG1QD3JFO9hJdZ+y4KutfdOQ==", + "peer": true, + "dependencies": { + "deep-equal": "^2.0.5" + } + }, + "node_modules/@testing-library/dom/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "peer": 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/@testing-library/dom/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==", + "peer": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@testing-library/dom/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==", + "peer": true + }, + "node_modules/@testing-library/dom/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==", + "peer": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@testing-library/dom/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==", + "peer": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@testing-library/jest-dom": { + "version": "5.17.0", + "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-5.17.0.tgz", + "integrity": "sha512-ynmNeT7asXyH3aSVv4vvX4Rb+0qjOhdNHnO/3vuZNqPmhDpV/+rCSGwQ7bLcmU2cJ4dvoheIO85LQj0IbJHEtg==", + "dependencies": { + "@adobe/css-tools": "^4.0.1", + "@babel/runtime": "^7.9.2", + "@types/testing-library__jest-dom": "^5.9.1", + "aria-query": "^5.0.0", + "chalk": "^3.0.0", + "css.escape": "^1.5.1", + "dom-accessibility-api": "^0.5.6", + "lodash": "^4.17.15", + "redent": "^3.0.0" + }, + "engines": { + "node": ">=8", + "npm": ">=6", + "yarn": ">=1" + } + }, + "node_modules/@testing-library/jest-dom/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@testing-library/jest-dom/node_modules/chalk": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@testing-library/jest-dom/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@testing-library/jest-dom/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==" + }, + "node_modules/@testing-library/jest-dom/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@testing-library/jest-dom/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@testing-library/react": { + "version": "13.4.0", + "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-13.4.0.tgz", + "integrity": "sha512-sXOGON+WNTh3MLE9rve97ftaZukN3oNf2KjDy7YTx6hcTO2uuLHuCGynMDhFwGw/jYf4OJ2Qk0i4i79qMNNkyw==", + "dependencies": { + "@babel/runtime": "^7.12.5", + "@testing-library/dom": "^8.5.0", + "@types/react-dom": "^18.0.0" + }, + "engines": { + "node": ">=12" + }, + "peerDependencies": { + "react": "^18.0.0", + "react-dom": "^18.0.0" + } + }, + "node_modules/@testing-library/react/node_modules/@testing-library/dom": { + "version": "8.20.1", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-8.20.1.tgz", + "integrity": "sha512-/DiOQ5xBxgdYRC8LNk7U+RWat0S3qRLeIw3ZIkMQ9kkVlRmwD/Eg8k8CqIpD6GW7u20JIUOfMKbxtiLutpjQ4g==", + "dependencies": { + "@babel/code-frame": "^7.10.4", + "@babel/runtime": "^7.12.5", + "@types/aria-query": "^5.0.1", + "aria-query": "5.1.3", + "chalk": "^4.1.0", + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "pretty-format": "^27.0.2" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@testing-library/react/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@testing-library/react/node_modules/aria-query": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.1.3.tgz", + "integrity": "sha512-R5iJ5lkuHybztUfuOAznmboyjWq8O6sqNqtK7CLOqdydi54VNbORp49mb14KbWgG1QD3JFO9hJdZ+y4KutfdOQ==", + "dependencies": { + "deep-equal": "^2.0.5" + } + }, + "node_modules/@testing-library/react/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/@testing-library/react/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@testing-library/react/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==" + }, + "node_modules/@testing-library/react/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@testing-library/react/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@testing-library/user-event": { + "version": "13.5.0", + "resolved": "https://registry.npmjs.org/@testing-library/user-event/-/user-event-13.5.0.tgz", + "integrity": "sha512-5Kwtbo3Y/NowpkbRuSepbyMFkZmHgD+vPzYB/RJ4oxt5Gj/avFFBYjhw27cqSVPVw/3a67NK1PbiIr9k4Gwmdg==", + "dependencies": { + "@babel/runtime": "^7.12.5" + }, + "engines": { + "node": ">=10", + "npm": ">=6" + }, + "peerDependencies": { + "@testing-library/dom": ">=7.21.4" + } + }, + "node_modules/@tootallnate/once": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", + "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/@trysound/sax": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.2.0.tgz", + "integrity": "sha512-L7z9BgrNEcYyUYtF+HaEfiS5ebkh9jXqbszz7pC0hRBPaatV0XjSD3+eHrpqFemQfgwiFF0QPIarnIihIDn7OA==", + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/@types/aria-query": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-5.0.1.tgz", + "integrity": "sha512-XTIieEY+gvJ39ChLcB4If5zHtPxt3Syj5rgZR+e1ctpmK8NjPf0zFqsz4JpLJT0xla9GFDKjy8Cpu331nrmE1Q==" + }, + "node_modules/@types/babel__core": { + "version": "7.20.1", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.1.tgz", + "integrity": "sha512-aACu/U/omhdk15O4Nfb+fHgH/z3QsfQzpnvRZhYhThms83ZnAOZz7zZAWO7mn2yyNQaA4xTO8GLK3uqFU4bYYw==", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.6.4", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.4.tgz", + "integrity": "sha512-tFkciB9j2K755yrTALxD44McOrk+gfpIpvC3sxHjRawj6PfnQxrse4Clq5y/Rq+G3mrBurMax/lG8Qn2t9mSsg==", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.1.tgz", + "integrity": "sha512-azBFKemX6kMg5Io+/rdGT0dkGreboUVR0Cdm3fz9QJWpaQGJRQXl7C+6hOTCZcMll7KFyEQpgbYI2lHdsS4U7g==", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.20.1", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.20.1.tgz", + "integrity": "sha512-MitHFXnhtgwsGZWtT68URpOvLN4EREih1u3QtQiN4VdAxWKRVvGCSvw/Qth0M0Qq3pJpnGOu5JaM/ydK7OGbqg==", + "dependencies": { + "@babel/types": "^7.20.7" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.2", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.2.tgz", + "integrity": "sha512-ALYone6pm6QmwZoAgeyNksccT9Q4AWZQ6PvfwR37GT6r6FWUPguq6sUmNGSMV2Wr761oQoBxwGGa6DR5o1DC9g==", + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/bonjour": { + "version": "3.5.10", + "resolved": "https://registry.npmjs.org/@types/bonjour/-/bonjour-3.5.10.tgz", + "integrity": "sha512-p7ienRMiS41Nu2/igbJxxLDWrSZ0WxM8UQgCeO9KhoVF7cOVFkrKsiDr1EsJIla8vV3oEEjGcz11jc5yimhzZw==", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.35", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.35.tgz", + "integrity": "sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ==", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/connect-history-api-fallback": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@types/connect-history-api-fallback/-/connect-history-api-fallback-1.5.0.tgz", + "integrity": "sha512-4x5FkPpLipqwthjPsF7ZRbOv3uoLUFkTA9G9v583qi4pACvq0uTELrB8OLUzPWUI4IJIyvM85vzkV1nyiI2Lig==", + "dependencies": { + "@types/express-serve-static-core": "*", + "@types/node": "*" + } + }, + "node_modules/@types/cookie": { + "version": "0.5.4", + "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.5.4.tgz", + "integrity": "sha512-7z/eR6O859gyWIAjuvBWFzNURmf2oPBmJlfVWkwehU5nzIyjwBsTh7WMmEEV4JFnHuQ3ex4oyTvfKzcyJVDBNA==" + }, + "node_modules/@types/eslint": { + "version": "8.44.2", + "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-8.44.2.tgz", + "integrity": "sha512-sdPRb9K6iL5XZOmBubg8yiFp5yS/JdUDQsq5e6h95km91MCYMuvp7mh1fjPEYUhvHepKpZOjnEaMBR4PxjWDzg==", + "dependencies": { + "@types/estree": "*", + "@types/json-schema": "*" + } + }, + "node_modules/@types/eslint-scope": { + "version": "3.7.4", + "resolved": "https://registry.npmjs.org/@types/eslint-scope/-/eslint-scope-3.7.4.tgz", + "integrity": "sha512-9K4zoImiZc3HlIp6AVUDE4CWYx22a+lhSZMYNpbjW04+YF0KWj4pJXnEMjdnFTiQibFFmElcsasJXDbdI/EPhA==", + "dependencies": { + "@types/eslint": "*", + "@types/estree": "*" + } + }, + "node_modules/@types/estree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.1.tgz", + "integrity": "sha512-LG4opVs2ANWZ1TJoKc937iMmNstM/d0ae1vNbnBvBhqCSezgVUOzcLCqbI5elV8Vy6WKwKjaqR+zO9VKirBBCA==" + }, + "node_modules/@types/express": { + "version": "4.17.17", + "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.17.tgz", + "integrity": "sha512-Q4FmmuLGBG58btUnfS1c1r/NQdlp3DMfGDGig8WhfpA2YRUtEkxAjkZb0yvplJGYdF1fsQ81iMDcH24sSCNC/Q==", + "dependencies": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^4.17.33", + "@types/qs": "*", + "@types/serve-static": "*" + } + }, + "node_modules/@types/express-serve-static-core": { + "version": "4.17.36", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.36.tgz", + "integrity": "sha512-zbivROJ0ZqLAtMzgzIUC4oNqDG9iF0lSsAqpOD9kbs5xcIM3dTiyuHvBc7R8MtWBp3AAWGaovJa+wzWPjLYW7Q==", + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*", + "@types/send": "*" + } + }, + "node_modules/@types/graceful-fs": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.6.tgz", + "integrity": "sha512-Sig0SNORX9fdW+bQuTEovKj3uHcUL6LQKbCrrqb1X7J6/ReAbhCXRAhc+SMejhLELFj2QcyuxmUooZ4bt5ReSw==", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/html-minifier-terser": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@types/html-minifier-terser/-/html-minifier-terser-6.1.0.tgz", + "integrity": "sha512-oh/6byDPnL1zeNXFrDXFLyZjkr1MsBG667IM792caf1L2UPOOMf65NFzjUH/ltyfwjAGfs1rsX1eftK0jC/KIg==" + }, + "node_modules/@types/http-errors": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.1.tgz", + "integrity": "sha512-/K3ds8TRAfBvi5vfjuz8y6+GiAYBZ0x4tXv1Av6CWBWn0IlADc+ZX9pMq7oU0fNQPnBwIZl3rmeLp6SBApbxSQ==" + }, + "node_modules/@types/http-proxy": { + "version": "1.17.11", + "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.11.tgz", + "integrity": "sha512-HC8G7c1WmaF2ekqpnFq626xd3Zz0uvaqFmBJNRZCGEZCXkvSdJoNFn/8Ygbd9fKNQj8UzLdCETaI0UWPAjK7IA==", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/istanbul-lib-coverage": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz", + "integrity": "sha512-z/QT1XN4K4KYuslS23k62yDIDLwLFkzxOuMplDtObz0+y7VqJCaO2o+SPwHCvLFZh7xazvvoor2tA/hPz9ee7g==" + }, + "node_modules/@types/istanbul-lib-report": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", + "integrity": "sha512-plGgXAPfVKFoYfa9NpYDAkseG+g6Jr294RqeqcqDixSbU34MZVJRi/P+7Y8GDpzkEwLaGZZOpKIEmeVZNtKsrg==", + "dependencies": { + "@types/istanbul-lib-coverage": "*" + } + }, + "node_modules/@types/istanbul-reports": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.1.tgz", + "integrity": "sha512-c3mAZEuK0lvBp8tmuL74XRKn1+y2dcwOUpH7x4WrF6gk1GIgiluDRgMYQtw2OFcBvAJWlt6ASU3tSqxp0Uu0Aw==", + "dependencies": { + "@types/istanbul-lib-report": "*" + } + }, + "node_modules/@types/jest": { + "version": "29.5.4", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.5.4.tgz", + "integrity": "sha512-PhglGmhWeD46FYOVLt3X7TiWjzwuVGW9wG/4qocPevXMjCmrIc5b6db9WjeGE4QYVpUAWMDv3v0IiBwObY289A==", + "dependencies": { + "expect": "^29.0.0", + "pretty-format": "^29.0.0" + } + }, + "node_modules/@types/jest/node_modules/@jest/schemas": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.6.3.tgz", + "integrity": "sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==", + "dependencies": { + "@sinclair/typebox": "^0.27.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/@jest/types": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.6.3.tgz", + "integrity": "sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw==", + "dependencies": { + "@jest/schemas": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^17.0.8", + "chalk": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/@sinclair/typebox": { + "version": "0.27.8", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.27.8.tgz", + "integrity": "sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==" + }, + "node_modules/@types/jest/node_modules/@types/yargs": { + "version": "17.0.24", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.24.tgz", + "integrity": "sha512-6i0aC7jV6QzQB8ne1joVZ0eSFIstHsCrobmOtghM11yGlH0j43FKL2UhWdELkyps0zuf7qVTUVCCR+tgSlyLLw==", + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/@types/jest/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@types/jest/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/@types/jest/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@types/jest/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==" + }, + "node_modules/@types/jest/node_modules/diff-sequences": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", + "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/expect": { + "version": "29.6.4", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.6.4.tgz", + "integrity": "sha512-F2W2UyQ8XYyftHT57dtfg8Ue3X5qLgm2sSug0ivvLRH/VKNRL/pDxg/TH7zVzbQB0tu80clNFy6LU7OS/VSEKA==", + "dependencies": { + "@jest/expect-utils": "^29.6.4", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.6.4", + "jest-message-util": "^29.6.3", + "jest-util": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@types/jest/node_modules/jest-diff": { + "version": "29.6.4", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.6.4.tgz", + "integrity": "sha512-9F48UxR9e4XOEZvoUXEHSWY4qC4zERJaOfrbBg9JpbJOO43R1vN76REt/aMGZoY6GD5g84nnJiBIVlscegefpw==", + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^29.6.3", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/jest-matcher-utils": { + "version": "29.6.4", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.6.4.tgz", + "integrity": "sha512-KSzwyzGvK4HcfnserYqJHYi7sZVqdREJ9DMPAKVbS98JsIAvumihaNUbjrWw0St7p9IY7A9UskCW5MYlGmBQFQ==", + "dependencies": { + "chalk": "^4.0.0", + "jest-diff": "^29.6.4", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/jest-message-util": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.6.3.tgz", + "integrity": "sha512-FtzaEEHzjDpQp51HX4UMkPZjy46ati4T5pEMyM6Ik48ztu4T9LQplZ6OsimHx7EuM9dfEh5HJa6D3trEftu3dA==", + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^29.6.3", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^29.6.3", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/jest-util": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.6.3.tgz", + "integrity": "sha512-QUjna/xSy4B32fzcKTSz1w7YYzgiHrjjJjevdRf61HYk998R5vVMMNmrHESYZVDS5DSWs+1srPLPKxXPkeSDOA==", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/pretty-format": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.6.3.tgz", + "integrity": "sha512-ZsBgjVhFAj5KeK+nHfF1305/By3lechHQSMWCTl8iHSbfOm2TN5nHEtFc/+W7fAyUeCs2n5iow72gld4gW0xDw==", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@types/jest/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@types/jest/node_modules/react-is": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.2.0.tgz", + "integrity": "sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==" + }, + "node_modules/@types/jest/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@types/json-schema": { + "version": "7.0.12", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.12.tgz", + "integrity": "sha512-Hr5Jfhc9eYOQNPYO5WLDq/n4jqijdHNlDXjuAQkkt+mWdQR+XJToOHrsD4cPaMXpn6KO7y2+wM8AZEs8VpBLVA==" + }, + "node_modules/@types/json5": { + "version": "0.0.29", + "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", + "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==" + }, + "node_modules/@types/mime": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.2.tgz", + "integrity": "sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw==" + }, + "node_modules/@types/node": { + "version": "20.5.6", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.5.6.tgz", + "integrity": "sha512-Gi5wRGPbbyOTX+4Y2iULQ27oUPrefaB0PxGQJnfyWN3kvEDGM3mIB5M/gQLmitZf7A9FmLeaqxD3L1CXpm3VKQ==" + }, + "node_modules/@types/parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==" + }, + "node_modules/@types/prettier": { + "version": "2.7.3", + "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.7.3.tgz", + "integrity": "sha512-+68kP9yzs4LMp7VNh8gdzMSPZFL44MLGqiHWvttYJe+6qnuVr4Ek9wSBQoveqY/r+LwjCcU29kNVkidwim+kYA==" + }, + "node_modules/@types/prop-types": { + "version": "15.7.5", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.5.tgz", + "integrity": "sha512-JCB8C6SnDoQf0cNycqd/35A7MjcnK+ZTqE7judS6o7utxUCg6imJg3QK2qzHKszlTjcj2cn+NwMB2i96ubpj7w==" + }, + "node_modules/@types/q": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@types/q/-/q-1.5.5.tgz", + "integrity": "sha512-L28j2FcJfSZOnL1WBjDYp2vUHCeIFlyYI/53EwD/rKUBQ7MtUUfbQWiyKJGpcnv4/WgrhWsFKrcPstcAt/J0tQ==" + }, + "node_modules/@types/qs": { + "version": "6.9.7", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz", + "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==" + }, + "node_modules/@types/range-parser": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.4.tgz", + "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw==" + }, + "node_modules/@types/react": { + "version": "18.2.21", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.21.tgz", + "integrity": "sha512-neFKG/sBAwGxHgXiIxnbm3/AAVQ/cMRS93hvBpg8xYRbeQSPVABp9U2bRnPf0iI4+Ucdv3plSxKK+3CW2ENJxA==", + "dependencies": { + "@types/prop-types": "*", + "@types/scheduler": "*", + "csstype": "^3.0.2" + } + }, + "node_modules/@types/react-dom": { + "version": "18.2.7", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.7.tgz", + "integrity": "sha512-GRaAEriuT4zp9N4p1i8BDBYmEyfo+xQ3yHjJU4eiK5NDa1RmUZG+unZABUTK4/Ox/M+GaHwb6Ow8rUITrtjszA==", + "dependencies": { + "@types/react": "*" + } + }, + "node_modules/@types/resolve": { + "version": "1.17.1", + "resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-1.17.1.tgz", + "integrity": "sha512-yy7HuzQhj0dhGpD8RLXSZWEkLsV9ibvxvi6EiJ3bkqLAO1RGo0WbkWQiwpRlSFymTJRz0d3k5LM3kkx8ArDbLw==", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/retry": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/@types/retry/-/retry-0.12.0.tgz", + "integrity": "sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA==" + }, + "node_modules/@types/scheduler": { + "version": "0.16.3", + "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.3.tgz", + "integrity": "sha512-5cJ8CB4yAx7BH1oMvdU0Jh9lrEXyPkar6F9G/ERswkCuvP4KQZfZkSjcMbAICCpQTN4OuZn8tz0HiKv9TGZgrQ==" + }, + "node_modules/@types/semver": { + "version": "7.5.0", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.0.tgz", + "integrity": "sha512-G8hZ6XJiHnuhQKR7ZmysCeJWE08o8T0AXtk5darsCaTVsYZhhgUrq53jizaR2FvsoeCwJhlmwTjkXBY5Pn/ZHw==" + }, + "node_modules/@types/send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.1.tgz", + "integrity": "sha512-Cwo8LE/0rnvX7kIIa3QHCkcuF21c05Ayb0ZfxPiv0W8VRiZiNW/WuRupHKpqqGVGf7SUA44QSOUKaEd9lIrd/Q==", + "dependencies": { + "@types/mime": "^1", + "@types/node": "*" + } + }, + "node_modules/@types/serve-index": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/@types/serve-index/-/serve-index-1.9.1.tgz", + "integrity": "sha512-d/Hs3nWDxNL2xAczmOVZNj92YZCS6RGxfBPjKzuu/XirCgXdpKEb88dYNbrYGint6IVWLNP+yonwVAuRC0T2Dg==", + "dependencies": { + "@types/express": "*" + } + }, + "node_modules/@types/serve-static": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.2.tgz", + "integrity": "sha512-J2LqtvFYCzaj8pVYKw8klQXrLLk7TBZmQ4ShlcdkELFKGwGMfevMLneMMRkMgZxotOD9wg497LpC7O8PcvAmfw==", + "dependencies": { + "@types/http-errors": "*", + "@types/mime": "*", + "@types/node": "*" + } + }, + "node_modules/@types/sockjs": { + "version": "0.3.33", + "resolved": "https://registry.npmjs.org/@types/sockjs/-/sockjs-0.3.33.tgz", + "integrity": "sha512-f0KEEe05NvUnat+boPTZ0dgaLZ4SfSouXUgv5noUiefG2ajgKjmETo9ZJyuqsl7dfl2aHlLJUiki6B4ZYldiiw==", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/stack-utils": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.1.tgz", + "integrity": "sha512-Hl219/BT5fLAaz6NDkSuhzasy49dwQS/DSdu4MdggFB8zcXv7vflBI3xp7FEmkmdDkBUI2bPUNeMttp2knYdxw==" + }, + "node_modules/@types/testing-library__jest-dom": { + "version": "5.14.9", + "resolved": "https://registry.npmjs.org/@types/testing-library__jest-dom/-/testing-library__jest-dom-5.14.9.tgz", + "integrity": "sha512-FSYhIjFlfOpGSRyVoMBMuS3ws5ehFQODymf3vlI7U1K8c7PHwWwFY7VREfmsuzHSOnoKs/9/Y983ayOs7eRzqw==", + "dependencies": { + "@types/jest": "*" + } + }, + "node_modules/@types/trusted-types": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.3.tgz", + "integrity": "sha512-NfQ4gyz38SL8sDNrSixxU2Os1a5xcdFxipAFxYEuLUlvU2uDwS4NUpsImcf1//SlWItCVMMLiylsxbmNMToV/g==" + }, + "node_modules/@types/ws": { + "version": "8.5.5", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.5.tgz", + "integrity": "sha512-lwhs8hktwxSjf9UaZ9tG5M03PGogvFaH8gUgLNbN9HKIg0dvv6q+gkSuJ8HN4/VbyxkuLzCjlN7GquQ0gUJfIg==", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/yargs": { + "version": "16.0.5", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-16.0.5.tgz", + "integrity": "sha512-AxO/ADJOBFJScHbWhq2xAhlWP24rY4aCEG/NFaMvbT3X2MgRsLjhjQwsn0Zi5zn0LG9jUhCCZMeX9Dkuw6k+vQ==", + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/@types/yargs-parser": { + "version": "21.0.0", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.0.tgz", + "integrity": "sha512-iO9ZQHkZxHn4mSakYV0vFHAVDyEOIJQrV2uZ06HxEPcx+mt8swXoZHIbaaJ2crJYFfErySgktuTZ3BeLz+XmFA==" + }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-5.62.0.tgz", + "integrity": "sha512-TiZzBSJja/LbhNPvk6yc0JrX9XqhQ0hdh6M2svYfsHGejaKFIAGd9MQ+ERIMzLGlN/kZoYIgdxFV0PuljTKXag==", + "dependencies": { + "@eslint-community/regexpp": "^4.4.0", + "@typescript-eslint/scope-manager": "5.62.0", + "@typescript-eslint/type-utils": "5.62.0", + "@typescript-eslint/utils": "5.62.0", + "debug": "^4.3.4", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "natural-compare-lite": "^1.4.0", + "semver": "^7.3.7", + "tsutils": "^3.21.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^5.0.0", + "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/experimental-utils": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-5.62.0.tgz", + "integrity": "sha512-RTXpeB3eMkpoclG3ZHft6vG/Z30azNHuqY6wKPBHlVMZFuEvrtlEDe8gMqDb+SO+9hjC/pLekeSCryf9vMZlCw==", + "dependencies": { + "@typescript-eslint/utils": "5.62.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-5.62.0.tgz", + "integrity": "sha512-VlJEV0fOQ7BExOsHYAGrgbEiZoi8D+Bl2+f6V2RrXerRSylnp+ZBHmPvaIa8cz0Ajx7WO7Z5RqfgYg7ED1nRhA==", + "dependencies": { + "@typescript-eslint/scope-manager": "5.62.0", + "@typescript-eslint/types": "5.62.0", + "@typescript-eslint/typescript-estree": "5.62.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-5.62.0.tgz", + "integrity": "sha512-VXuvVvZeQCQb5Zgf4HAxc04q5j+WrNAtNh9OwCsCgpKqESMTu3tF/jhZ3xG6T4NZwWl65Bg8KuS2uEvhSfLl0w==", + "dependencies": { + "@typescript-eslint/types": "5.62.0", + "@typescript-eslint/visitor-keys": "5.62.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-5.62.0.tgz", + "integrity": "sha512-xsSQreu+VnfbqQpW5vnCJdq1Z3Q0U31qiWmRhr98ONQmcp/yhiPJFPq8MXiJVLiksmOKSjIldZzkebzHuCGzew==", + "dependencies": { + "@typescript-eslint/typescript-estree": "5.62.0", + "@typescript-eslint/utils": "5.62.0", + "debug": "^4.3.4", + "tsutils": "^3.21.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "*" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/types": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-5.62.0.tgz", + "integrity": "sha512-87NVngcbVXUahrRTqIK27gD2t5Cu1yuCXxbLcFtCzZGlfyVWWh8mLHkoxzjsB6DDNnvdL+fW8MiwPEJyGJQDgQ==", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-5.62.0.tgz", + "integrity": "sha512-CmcQ6uY7b9y694lKdRB8FEel7JbU/40iSAPomu++SjLMntB+2Leay2LO6i8VnJk58MtE9/nQSFIH6jpyRWyYzA==", + "dependencies": { + "@typescript-eslint/types": "5.62.0", + "@typescript-eslint/visitor-keys": "5.62.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "semver": "^7.3.7", + "tsutils": "^3.21.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-5.62.0.tgz", + "integrity": "sha512-n8oxjeb5aIbPFEtmQxQYOLI0i9n5ySBEY/ZEHHZqKQSFnxio1rv6dthascc9dLuwrL0RC5mPCxB7vnAVGAYWAQ==", + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@types/json-schema": "^7.0.9", + "@types/semver": "^7.3.12", + "@typescript-eslint/scope-manager": "5.62.0", + "@typescript-eslint/types": "5.62.0", + "@typescript-eslint/typescript-estree": "5.62.0", + "eslint-scope": "^5.1.1", + "semver": "^7.3.7" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "5.62.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-5.62.0.tgz", + "integrity": "sha512-07ny+LHRzQXepkGg6w0mFY41fVUNBrL2Roj/++7V1txKugfjm/Ci/qSND03r2RhlJhJYMcTn9AhhSSqQp0Ysyw==", + "dependencies": { + "@typescript-eslint/types": "5.62.0", + "eslint-visitor-keys": "^3.3.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@webassemblyjs/ast": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.11.6.tgz", + "integrity": "sha512-IN1xI7PwOvLPgjcf180gC1bqn3q/QaOCwYUahIOhbYUu8KA/3tw2RT/T0Gidi1l7Hhj5D/INhJxiICObqpMu4Q==", + "dependencies": { + "@webassemblyjs/helper-numbers": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6" + } + }, + "node_modules/@webassemblyjs/floating-point-hex-parser": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.11.6.tgz", + "integrity": "sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw==" + }, + "node_modules/@webassemblyjs/helper-api-error": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.11.6.tgz", + "integrity": "sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q==" + }, + "node_modules/@webassemblyjs/helper-buffer": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.11.6.tgz", + "integrity": "sha512-z3nFzdcp1mb8nEOFFk8DrYLpHvhKC3grJD2ardfKOzmbmJvEf/tPIqCY+sNcwZIY8ZD7IkB2l7/pqhUhqm7hLA==" + }, + "node_modules/@webassemblyjs/helper-numbers": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-numbers/-/helper-numbers-1.11.6.tgz", + "integrity": "sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g==", + "dependencies": { + "@webassemblyjs/floating-point-hex-parser": "1.11.6", + "@webassemblyjs/helper-api-error": "1.11.6", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/helper-wasm-bytecode": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.11.6.tgz", + "integrity": "sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA==" + }, + "node_modules/@webassemblyjs/helper-wasm-section": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.11.6.tgz", + "integrity": "sha512-LPpZbSOwTpEC2cgn4hTydySy1Ke+XEu+ETXuoyvuyezHO3Kjdu90KK95Sh9xTbmjrCsUwvWwCOQQNta37VrS9g==", + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-buffer": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/wasm-gen": "1.11.6" + } + }, + "node_modules/@webassemblyjs/ieee754": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.11.6.tgz", + "integrity": "sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg==", + "dependencies": { + "@xtuc/ieee754": "^1.2.0" + } + }, + "node_modules/@webassemblyjs/leb128": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.11.6.tgz", + "integrity": "sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ==", + "dependencies": { + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/utf8": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.11.6.tgz", + "integrity": "sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA==" + }, + "node_modules/@webassemblyjs/wasm-edit": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.11.6.tgz", + "integrity": "sha512-Ybn2I6fnfIGuCR+Faaz7YcvtBKxvoLV3Lebn1tM4o/IAJzmi9AWYIPWpyBfU8cC+JxAO57bk4+zdsTjJR+VTOw==", + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-buffer": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/helper-wasm-section": "1.11.6", + "@webassemblyjs/wasm-gen": "1.11.6", + "@webassemblyjs/wasm-opt": "1.11.6", + "@webassemblyjs/wasm-parser": "1.11.6", + "@webassemblyjs/wast-printer": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wasm-gen": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.11.6.tgz", + "integrity": "sha512-3XOqkZP/y6B4F0PBAXvI1/bky7GryoogUtfwExeP/v7Nzwo1QLcq5oQmpKlftZLbT+ERUOAZVQjuNVak6UXjPA==", + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/ieee754": "1.11.6", + "@webassemblyjs/leb128": "1.11.6", + "@webassemblyjs/utf8": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wasm-opt": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.11.6.tgz", + "integrity": "sha512-cOrKuLRE7PCe6AsOVl7WasYf3wbSo4CeOk6PkrjS7g57MFfVUF9u6ysQBBODX0LdgSvQqRiGz3CXvIDKcPNy4g==", + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-buffer": "1.11.6", + "@webassemblyjs/wasm-gen": "1.11.6", + "@webassemblyjs/wasm-parser": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wasm-parser": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.11.6.tgz", + "integrity": "sha512-6ZwPeGzMJM3Dqp3hCsLgESxBGtT/OeCvCZ4TA1JUPYgmhAx38tTPR9JaKy0S5H3evQpO/h2uWs2j6Yc/fjkpTQ==", + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-api-error": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/ieee754": "1.11.6", + "@webassemblyjs/leb128": "1.11.6", + "@webassemblyjs/utf8": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wast-printer": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.11.6.tgz", + "integrity": "sha512-JM7AhRcE+yW2GWYaKeHL5vt4xqee5N2WcezptmgyhNS+ScggqcT1OtXykhAb13Sn5Yas0j2uv9tHgrjwvzAP4A==", + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@xtuc/ieee754": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", + "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==" + }, + "node_modules/@xtuc/long": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz", + "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==" + }, + "node_modules/abab": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.6.tgz", + "integrity": "sha512-j2afSsaIENvHZN2B8GOpF566vZ5WVk5opAiMTvWgaQT8DkbOqsTfvNAvHoRGU2zzP8cPoqys+xHTRDWW8L+/BA==" + }, + "node_modules/accepts": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", + "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", + "dependencies": { + "mime-types": "~2.1.34", + "negotiator": "0.6.3" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/acorn": { + "version": "8.10.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.10.0.tgz", + "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==", + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-globals": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-6.0.0.tgz", + "integrity": "sha512-ZQl7LOWaF5ePqqcX4hLuv/bLXYQNfNWw2c0/yX/TsPRKamzHcTGQnlCjHT3TsmkOUVEPS3crCxiPfdzE/Trlhg==", + "dependencies": { + "acorn": "^7.1.1", + "acorn-walk": "^7.1.1" + } + }, + "node_modules/acorn-globals/node_modules/acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-import-assertions": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/acorn-import-assertions/-/acorn-import-assertions-1.9.0.tgz", + "integrity": "sha512-cmMwop9x+8KFhxvKrKfPYmN6/pKTYYHBqLa0DfvVZcKMJWNyWLnaqND7dx/qn66R7ewM1UX5XMaDVP5wlVTaVA==", + "peerDependencies": { + "acorn": "^8" + } + }, + "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==", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/acorn-walk": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz", + "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/address": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/address/-/address-1.2.2.tgz", + "integrity": "sha512-4B/qKCfeE/ODUaAUpSwfzazo5x29WD4r3vXiWsB7I2mSDAihwEqKO+g8GELZUQSSAo5e1XTYh3ZVfLyxBc12nA==", + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/adjust-sourcemap-loader": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/adjust-sourcemap-loader/-/adjust-sourcemap-loader-4.0.0.tgz", + "integrity": "sha512-OXwN5b9pCUXNQHJpwwD2qP40byEmSgzj8B4ydSN0uMNYWiFmJ6x6KwUllMmfk8Rwu/HJDFR7U8ubsWBoN0Xp0A==", + "dependencies": { + "loader-utils": "^2.0.0", + "regex-parser": "^2.2.11" + }, + "engines": { + "node": ">=8.9" + } + }, + "node_modules/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.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==", + "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/ajv-formats": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ajv-formats/-/ajv-formats-2.1.1.tgz", + "integrity": "sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==", + "dependencies": { + "ajv": "^8.0.0" + }, + "peerDependencies": { + "ajv": "^8.0.0" + }, + "peerDependenciesMeta": { + "ajv": { + "optional": true + } + } + }, + "node_modules/ajv-formats/node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ajv-formats/node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/ajv-keywords": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "peerDependencies": { + "ajv": "^6.9.1" + } + }, + "node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-html-community": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/ansi-html-community/-/ansi-html-community-0.0.8.tgz", + "integrity": "sha512-1APHAyr3+PCamwNw3bXCPp4HFLONZt/yIH0sZp0/469KWNTEy+qN5jQ3GVX6DMZ1UXAi34yVwtTeaG/HpBuuzw==", + "engines": [ + "node >= 0.8.0" + ], + "bin": { + "ansi-html": "bin/ansi-html" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/any-promise": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz", + "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==" + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/arg": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz", + "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==" + }, + "node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/aria-query": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", + "dependencies": { + "dequal": "^2.0.3" + } + }, + "node_modules/array-buffer-byte-length": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.0.tgz", + "integrity": "sha512-LPuwb2P+NrQw3XhxGc36+XSvuBPopovXYTR9Ew++Du9Yb/bx5AzBfrIsBoj0EZUifjQU+sHL21sseZ3jerWO/A==", + "dependencies": { + "call-bind": "^1.0.2", + "is-array-buffer": "^3.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array-flatten": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-2.1.2.tgz", + "integrity": "sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ==" + }, + "node_modules/array-includes": { + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.6.tgz", + "integrity": "sha512-sgTbLvL6cNnw24FnbaDyjmvddQ2ML8arZsgaJhoABMoplz/4QRhtrYS+alr1BUM1Bwp6dhx8vVCBSLG+StwOFw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "get-intrinsic": "^1.1.3", + "is-string": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "engines": { + "node": ">=8" + } + }, + "node_modules/array.prototype.findlastindex": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.2.tgz", + "integrity": "sha512-tb5thFFlUcp7NdNF6/MpDk/1r/4awWG1FIz3YqDf+/zJSTezBb+/5WViH41obXULHVpDzoiCLpJ/ZO9YbJMsdw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "es-shim-unscopables": "^1.0.0", + "get-intrinsic": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.flat": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.1.tgz", + "integrity": "sha512-roTU0KWIOmJ4DRLmwKd19Otg0/mT3qPNt0Qb3GWW8iObuZXxrjB/pzn0R3hqpRSWg4HCwqx+0vwOnWnvlOyeIA==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "es-shim-unscopables": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.flatmap": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.1.tgz", + "integrity": "sha512-8UGn9O1FDVvMNB0UlLv4voxRMze7+FpHyF5mSMRjWHUMlpoDViniy05870VlxhfgTnLbpuwTzvD76MTtWxB/mQ==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "es-shim-unscopables": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.reduce": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/array.prototype.reduce/-/array.prototype.reduce-1.0.5.tgz", + "integrity": "sha512-kDdugMl7id9COE8R7MHF5jWk7Dqt/fs4Pv+JXoICnYwqpjjjbUurz6w5fT5IG6brLdJhv6/VoHB0H7oyIBXd+Q==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "es-array-method-boxes-properly": "^1.0.0", + "is-string": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.tosorted": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array.prototype.tosorted/-/array.prototype.tosorted-1.1.1.tgz", + "integrity": "sha512-pZYPXPRl2PqWcsUs6LOMn+1f1532nEoPTYowBtqLwAW+W8vSVhkIGnmOX1t/UQjD6YGI0vcD2B1U7ZFGQH9jnQ==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "es-shim-unscopables": "^1.0.0", + "get-intrinsic": "^1.1.3" + } + }, + "node_modules/arraybuffer.prototype.slice": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.1.tgz", + "integrity": "sha512-09x0ZWFEjj4WD8PDbykUwo3t9arLn8NIzmmYEJFpYekOAQjpkGSyrQhNoRTcwwcFRu+ycWF78QZ63oWTqSjBcw==", + "dependencies": { + "array-buffer-byte-length": "^1.0.0", + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "get-intrinsic": "^1.2.1", + "is-array-buffer": "^3.0.2", + "is-shared-array-buffer": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/asap": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", + "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==" + }, + "node_modules/ast-types-flow": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.7.tgz", + "integrity": "sha512-eBvWn1lvIApYMhzQMsu9ciLfkBY499mFZlNqG+/9WR7PVlroQw0vG30cOQQbaKz3sCEc44TAOu2ykzqXSNnwag==" + }, + "node_modules/async": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.4.tgz", + "integrity": "sha512-iAB+JbDEGXhyIUavoDl9WP/Jj106Kz9DEn1DPgYw5ruDn0e3Wgi3sKFm55sASdGBNOQB8F59d9qQ7deqrHA8wQ==" + }, + "node_modules/asynciterator.prototype": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/asynciterator.prototype/-/asynciterator.prototype-1.0.0.tgz", + "integrity": "sha512-wwHYEIS0Q80f5mosx3L/dfG5t5rjEa9Ft51GTaNt862EnpyGHpgz2RkZvLPp1oF5TnAiTohkEKVEu8pQPJI7Vg==", + "dependencies": { + "has-symbols": "^1.0.3" + } + }, + "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/at-least-node": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/at-least-node/-/at-least-node-1.0.0.tgz", + "integrity": "sha512-+q/t7Ekv1EDY2l6Gda6LLiX14rU9TV20Wa3ofeQmwPFZbOMo9DXrLbOjFaaclkXKWidIaopwAObQDqwWtGUjqg==", + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/autoprefixer": { + "version": "10.4.15", + "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.15.tgz", + "integrity": "sha512-KCuPB8ZCIqFdA4HwKXsvz7j6gvSDNhDP7WnUjBleRkKjPdvCmHFuQ77ocavI8FT6NdvlBnE2UFr2H4Mycn8Vew==", + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/autoprefixer" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "browserslist": "^4.21.10", + "caniuse-lite": "^1.0.30001520", + "fraction.js": "^4.2.0", + "normalize-range": "^0.1.2", + "picocolors": "^1.0.0", + "postcss-value-parser": "^4.2.0" + }, + "bin": { + "autoprefixer": "bin/autoprefixer" + }, + "engines": { + "node": "^10 || ^12 || >=14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/available-typed-arrays": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz", + "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/axe-core": { + "version": "4.7.2", + "resolved": "https://registry.npmjs.org/axe-core/-/axe-core-4.7.2.tgz", + "integrity": "sha512-zIURGIS1E1Q4pcrMjp+nnEh+16G56eG/MUllJH8yEvw7asDo7Ac9uhC9KIH5jzpITueEZolfYglnCGIuSBz39g==", + "engines": { + "node": ">=4" + } + }, + "node_modules/axios": { + "version": "1.6.7", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.7.tgz", + "integrity": "sha512-/hDJGff6/c7u0hDkvkGxR/oy6CbCs8ziCsC7SqmhjfozqiJGc8Z11wrv9z9lYfY4K8l+H9TpjcMDX0xOZmx+RA==", + "dependencies": { + "follow-redirects": "^1.15.4", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/axios/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/axobject-query": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-3.2.1.tgz", + "integrity": "sha512-jsyHu61e6N4Vbz/v18DHwWYKK0bSWLqn47eeDSKPB7m8tqMHF9YJ+mhIk2lVteyZrY8tnSj/jHOv4YiTCuCJgg==", + "dependencies": { + "dequal": "^2.0.3" + } + }, + "node_modules/babel-jest": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-27.5.1.tgz", + "integrity": "sha512-cdQ5dXjGRd0IBRATiQ4mZGlGlRE8kJpjPOixdNRdT+m3UcNqmYWN6rK6nvtXYfY3D76cb8s/O1Ss8ea24PIwcg==", + "dependencies": { + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/babel__core": "^7.1.14", + "babel-plugin-istanbul": "^6.1.1", + "babel-preset-jest": "^27.5.1", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "slash": "^3.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.8.0" + } + }, + "node_modules/babel-jest/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/babel-jest/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/babel-jest/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/babel-jest/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==" + }, + "node_modules/babel-jest/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-jest/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.3.0.tgz", + "integrity": "sha512-H8SvsMF+m9t15HNLMipppzkC+Y2Yq+v3SonZyU70RBL/h1gxPkH08Ot8pEE9Z4Kd+czyWJClmFS8qzIP9OZ04Q==", + "dependencies": { + "find-cache-dir": "^3.3.1", + "loader-utils": "^2.0.0", + "make-dir": "^3.1.0", + "schema-utils": "^2.6.5" + }, + "engines": { + "node": ">= 8.9" + }, + "peerDependencies": { + "@babel/core": "^7.0.0", + "webpack": ">=2" + } + }, + "node_modules/babel-loader/node_modules/schema-utils": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-2.7.1.tgz", + "integrity": "sha512-SHiNtMOUGWBQJwzISiVYKu82GiV4QYGePp3odlY1tuKO7gPtphAT5R/py0fA6xtbgLL/RvtJZnU9b8s0F1q0Xg==", + "dependencies": { + "@types/json-schema": "^7.0.5", + "ajv": "^6.12.4", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 8.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/babel-plugin-istanbul": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", + "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^5.0.4", + "test-exclude": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-jest-hoist": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-27.5.1.tgz", + "integrity": "sha512-50wCwD5EMNW4aRpOwtqzyZHIewTYNxLA4nhB+09d8BIssfNfzBRhkBIHiaPv1Si226TQSvp8gxAJm2iY2qs2hQ==", + "dependencies": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.0.0", + "@types/babel__traverse": "^7.0.6" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/babel-plugin-macros": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/babel-plugin-macros/-/babel-plugin-macros-3.1.0.tgz", + "integrity": "sha512-Cg7TFGpIr01vOQNODXOOaGz2NpCU5gl8x1qJFbb6hbZxR7XrcE2vtbAsTAbJ7/xwJtUuJEw8K8Zr/AE0LHlesg==", + "dependencies": { + "@babel/runtime": "^7.12.5", + "cosmiconfig": "^7.0.0", + "resolve": "^1.19.0" + }, + "engines": { + "node": ">=10", + "npm": ">=6" + } + }, + "node_modules/babel-plugin-named-asset-import": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/babel-plugin-named-asset-import/-/babel-plugin-named-asset-import-0.3.8.tgz", + "integrity": "sha512-WXiAc++qo7XcJ1ZnTYGtLxmBCVbddAml3CEXgWaBzNzLNoxtQ8AiGEFDMOhot9XjTCQbvP5E77Fj9Gk924f00Q==", + "peerDependencies": { + "@babel/core": "^7.1.0" + } + }, + "node_modules/babel-plugin-polyfill-corejs2": { + "version": "0.4.5", + "resolved": "https://registry.npmjs.org/babel-plugin-polyfill-corejs2/-/babel-plugin-polyfill-corejs2-0.4.5.tgz", + "integrity": "sha512-19hwUH5FKl49JEsvyTcoHakh6BE0wgXLLptIyKZ3PijHc/Ci521wygORCUCCred+E/twuqRyAkE02BAWPmsHOg==", + "dependencies": { + "@babel/compat-data": "^7.22.6", + "@babel/helper-define-polyfill-provider": "^0.4.2", + "semver": "^6.3.1" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/babel-plugin-polyfill-corejs2/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/babel-plugin-polyfill-corejs3": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/babel-plugin-polyfill-corejs3/-/babel-plugin-polyfill-corejs3-0.8.3.tgz", + "integrity": "sha512-z41XaniZL26WLrvjy7soabMXrfPWARN25PZoriDEiLMxAp50AUW3t35BGQUMg5xK3UrpVTtagIDklxYa+MhiNA==", + "dependencies": { + "@babel/helper-define-polyfill-provider": "^0.4.2", + "core-js-compat": "^3.31.0" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/babel-plugin-polyfill-regenerator": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/babel-plugin-polyfill-regenerator/-/babel-plugin-polyfill-regenerator-0.5.2.tgz", + "integrity": "sha512-tAlOptU0Xj34V1Y2PNTL4Y0FOJMDB6bZmoW39FeCQIhigGLkqu3Fj6uiXpxIf6Ij274ENdYx64y6Au+ZKlb1IA==", + "dependencies": { + "@babel/helper-define-polyfill-provider": "^0.4.2" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/babel-plugin-transform-react-remove-prop-types": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-react-remove-prop-types/-/babel-plugin-transform-react-remove-prop-types-0.4.24.tgz", + "integrity": "sha512-eqj0hVcJUR57/Ug2zE1Yswsw4LhuqqHhD+8v120T1cl3kjg76QwtyBrdIk4WVwK+lAhBJVYCd/v+4nc4y+8JsA==" + }, + "node_modules/babel-preset-current-node-syntax": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.1.tgz", + "integrity": "sha512-M7LQ0bxarkxQoN+vz5aJPsLBn77n8QgTFmo8WK0/44auK2xlCXrYcUxHFxgU7qW5Yzw/CjmLRK2uJzaCd7LvqQ==", + "dependencies": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.8.3", + "@babel/plugin-syntax-import-meta": "^7.8.3", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.8.3", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.8.3", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-top-level-await": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/babel-preset-jest": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-27.5.1.tgz", + "integrity": "sha512-Nptf2FzlPCWYuJg41HBqXVT8ym6bXOevuCTbhxlUpjwtysGaIWFvDEjp4y+G7fl13FgOdjs7P/DmErqH7da0Ag==", + "dependencies": { + "babel-plugin-jest-hoist": "^27.5.1", + "babel-preset-current-node-syntax": "^1.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/babel-preset-react-app": { + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/babel-preset-react-app/-/babel-preset-react-app-10.0.1.tgz", + "integrity": "sha512-b0D9IZ1WhhCWkrTXyFuIIgqGzSkRIH5D5AmB0bXbzYAB1OBAwHcUeyWW2LorutLWF5btNo/N7r/cIdmvvKJlYg==", + "dependencies": { + "@babel/core": "^7.16.0", + "@babel/plugin-proposal-class-properties": "^7.16.0", + "@babel/plugin-proposal-decorators": "^7.16.4", + "@babel/plugin-proposal-nullish-coalescing-operator": "^7.16.0", + "@babel/plugin-proposal-numeric-separator": "^7.16.0", + "@babel/plugin-proposal-optional-chaining": "^7.16.0", + "@babel/plugin-proposal-private-methods": "^7.16.0", + "@babel/plugin-transform-flow-strip-types": "^7.16.0", + "@babel/plugin-transform-react-display-name": "^7.16.0", + "@babel/plugin-transform-runtime": "^7.16.4", + "@babel/preset-env": "^7.16.4", + "@babel/preset-react": "^7.16.0", + "@babel/preset-typescript": "^7.16.0", + "@babel/runtime": "^7.16.3", + "babel-plugin-macros": "^3.1.0", + "babel-plugin-transform-react-remove-prop-types": "^0.4.24" + } + }, + "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==" + }, + "node_modules/batch": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/batch/-/batch-0.6.1.tgz", + "integrity": "sha512-x+VAiMRL6UPkx+kudNvxTl6hB2XNNCG2r+7wixVfIYwu/2HKRXimwQyaumLjMveWvT2Hkd/cAJw+QBMfJ/EKVw==" + }, + "node_modules/bfj": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/bfj/-/bfj-7.0.2.tgz", + "integrity": "sha512-+e/UqUzwmzJamNF50tBV6tZPTORow7gQ96iFow+8b562OdMpEK0BcJEq2OSPEDmAbSMBQ7PKZ87ubFkgxpYWgw==", + "dependencies": { + "bluebird": "^3.5.5", + "check-types": "^11.1.1", + "hoopy": "^0.1.4", + "tryer": "^1.0.1" + }, + "engines": { + "node": ">= 8.0.0" + } + }, + "node_modules/big.js": { + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", + "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", + "engines": { + "node": "*" + } + }, + "node_modules/binary-extensions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", + "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "engines": { + "node": ">=8" + } + }, + "node_modules/bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==" + }, + "node_modules/body-parser": { + "version": "1.20.1", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.1.tgz", + "integrity": "sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw==", + "dependencies": { + "bytes": "3.1.2", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "on-finished": "2.4.1", + "qs": "6.11.0", + "raw-body": "2.5.1", + "type-is": "~1.6.18", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/body-parser/node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/body-parser/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/body-parser/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/body-parser/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/bonjour-service": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/bonjour-service/-/bonjour-service-1.1.1.tgz", + "integrity": "sha512-Z/5lQRMOG9k7W+FkeGTNjh7htqn/2LMnfOvBZ8pynNZCM9MwkQkI3zeI4oz09uWdcgmgHugVvBqxGg4VQJ5PCg==", + "dependencies": { + "array-flatten": "^2.1.2", + "dns-equal": "^1.0.0", + "fast-deep-equal": "^3.1.3", + "multicast-dns": "^7.2.5" + } + }, + "node_modules/boolbase": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", + "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==" + }, + "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==", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browser-process-hrtime": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz", + "integrity": "sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==" + }, + "node_modules/browserslist": { + "version": "4.21.10", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.21.10.tgz", + "integrity": "sha512-bipEBdZfVH5/pwrvqc+Ub0kUPVfGUhlKxbvfD+z1BDnPEO/X98ruXGA1WP5ASpAFKan7Qr6j736IacbZQuAlKQ==", + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "caniuse-lite": "^1.0.30001517", + "electron-to-chromium": "^1.4.477", + "node-releases": "^2.0.13", + "update-browserslist-db": "^1.0.11" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dependencies": { + "node-int64": "^0.4.0" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" + }, + "node_modules/builtin-modules": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-3.3.0.tgz", + "integrity": "sha512-zhaCDicdLuWN5UbN5IMnFqNMhNfo919sH85y2/ea+5Yg9TsTkeZxpL+JLbp6cgYFS4sRLp3YV4S6yDuqVWHYOw==", + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/bytes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", + "integrity": "sha512-pMhOfFDPiv9t5jjIXkHosWmkSyQbvsgEVNkz0ERHbuLh2T/7j4Mqqpz523Fe8MVY89KC6Sh/QfS2sM+SjgFDcw==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dependencies": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/camel-case": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-4.1.2.tgz", + "integrity": "sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw==", + "dependencies": { + "pascal-case": "^3.1.2", + "tslib": "^2.0.3" + } + }, + "node_modules/camelcase": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/camelcase-css": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz", + "integrity": "sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/caniuse-api": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/caniuse-api/-/caniuse-api-3.0.0.tgz", + "integrity": "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==", + "dependencies": { + "browserslist": "^4.0.0", + "caniuse-lite": "^1.0.0", + "lodash.memoize": "^4.1.2", + "lodash.uniq": "^4.5.0" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001523", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001523.tgz", + "integrity": "sha512-I5q5cisATTPZ1mc588Z//pj/Ox80ERYDfR71YnvY7raS/NOk8xXlZcB0sF7JdqaV//kOaa6aus7lRfpdnt1eBA==", + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ] + }, + "node_modules/case-sensitive-paths-webpack-plugin": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/case-sensitive-paths-webpack-plugin/-/case-sensitive-paths-webpack-plugin-2.4.0.tgz", + "integrity": "sha512-roIFONhcxog0JSSWbvVAh3OocukmSgpqOH6YpMkCvav/ySIV3JKg4Dc8vYtQjYi/UxpNE36r/9v+VqTQqgkYmw==", + "engines": { + "node": ">=4" + } + }, + "node_modules/chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "engines": { + "node": ">=10" + } + }, + "node_modules/check-types": { + "version": "11.2.2", + "resolved": "https://registry.npmjs.org/check-types/-/check-types-11.2.2.tgz", + "integrity": "sha512-HBiYvXvn9Z70Z88XKjz3AEKd4HJhBXsa3j7xFnITAzoS8+q6eIGi8qDB8FKPBAjtuxjI/zFpwuiCb8oDtKOYrA==" + }, + "node_modules/chokidar": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", + "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ], + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/chokidar/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==", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/chrome-trace-event": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz", + "integrity": "sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg==", + "engines": { + "node": ">=6.0" + } + }, + "node_modules/ci-info": { + "version": "3.8.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.8.0.tgz", + "integrity": "sha512-eXTggHWSooYhq49F2opQhuHWgzucfF2YgODK4e1566GQs5BIfP30B0oenwBJHfWxAs2fyPB1s7Mg949zLf61Yw==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/sibiraj-s" + } + ], + "engines": { + "node": ">=8" + } + }, + "node_modules/cjs-module-lexer": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.2.3.tgz", + "integrity": "sha512-0TNiGstbQmCFwt4akjjBg5pLRTSyj/PkWQ1ZoO2zntmg9yLqSRxwEa4iCfQLGjqhiqBfOJa7W/E8wfGrTDmlZQ==" + }, + "node_modules/clean-css": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-5.3.2.tgz", + "integrity": "sha512-JVJbM+f3d3Q704rF4bqQ5UUyTtuJ0JRKNbTKVEeujCCBoMdkEi+V+e8oktO9qGQNSvHrFTM6JZRXrUvGR1czww==", + "dependencies": { + "source-map": "~0.6.0" + }, + "engines": { + "node": ">= 10.0" + } + }, + "node_modules/clean-css/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cliui": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^7.0.0" + } + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/coa": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/coa/-/coa-2.0.2.tgz", + "integrity": "sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA==", + "dependencies": { + "@types/q": "^1.5.1", + "chalk": "^2.4.1", + "q": "^1.1.2" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/collect-v8-coverage": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.2.tgz", + "integrity": "sha512-lHl4d5/ONEbLlJvaJNtsF/Lz+WvB07u2ycqTYbdrq7UypDXailES4valYb2eWiJFxZlVmpGekfqoxQhzyFdT4Q==" + }, + "node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==" + }, + "node_modules/colord": { + "version": "2.9.3", + "resolved": "https://registry.npmjs.org/colord/-/colord-2.9.3.tgz", + "integrity": "sha512-jeC1axXpnb0/2nn/Y1LPuLdgXBLH7aDcHu4KEKfqw3CUhX7ZpfBSlPKyqXE6btIgEzfWtrX3/tyBCaCvXvMkOw==" + }, + "node_modules/colorette": { + "version": "2.0.20", + "resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.20.tgz", + "integrity": "sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==" + }, + "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/commander": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", + "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", + "engines": { + "node": ">= 12" + } + }, + "node_modules/common-path-prefix": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/common-path-prefix/-/common-path-prefix-3.0.0.tgz", + "integrity": "sha512-QE33hToZseCH3jS0qN96O/bSh3kaw/h+Tq7ngyY9eWDUnTlTNUyqfqvCXioLe5Na5jFsL78ra/wuBU4iuEgd4w==" + }, + "node_modules/common-tags": { + "version": "1.8.2", + "resolved": "https://registry.npmjs.org/common-tags/-/common-tags-1.8.2.tgz", + "integrity": "sha512-gk/Z852D2Wtb//0I+kRFNKKE9dIIVirjoqPoA1wJU+XePVXZfGeBpk45+A1rKO4Q43prqWBNY/MiIeRLbPWUaA==", + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/commondir": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", + "integrity": "sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==" + }, + "node_modules/compressible": { + "version": "2.0.18", + "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.18.tgz", + "integrity": "sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==", + "dependencies": { + "mime-db": ">= 1.43.0 < 2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/compression": { + "version": "1.7.4", + "resolved": "https://registry.npmjs.org/compression/-/compression-1.7.4.tgz", + "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", + "dependencies": { + "accepts": "~1.3.5", + "bytes": "3.0.0", + "compressible": "~2.0.16", + "debug": "2.6.9", + "on-headers": "~1.0.2", + "safe-buffer": "5.1.2", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/compression/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/compression/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/compression/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" + }, + "node_modules/confusing-browser-globals": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/confusing-browser-globals/-/confusing-browser-globals-1.0.11.tgz", + "integrity": "sha512-JsPKdmh8ZkmnHxDk55FZ1TqVLvEQTvoByJZRN9jzI0UjxK/QgAmsphz7PGtqgPieQZ/CQcHWXCR7ATDNhGe+YA==" + }, + "node_modules/connect-history-api-fallback": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/connect-history-api-fallback/-/connect-history-api-fallback-2.0.0.tgz", + "integrity": "sha512-U73+6lQFmfiNPrYbXqr6kZ1i1wiRqXnp2nhMsINseWXO8lDau0LGEffJ8kQi4EjLZympVgRdvqjAgiZ1tgzDDA==", + "engines": { + "node": ">=0.8" + } + }, + "node_modules/content-disposition": { + "version": "0.5.4", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", + "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", + "dependencies": { + "safe-buffer": "5.2.1" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", + "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/convert-source-map": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.9.0.tgz", + "integrity": "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==" + }, + "node_modules/cookie": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz", + "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==" + }, + "node_modules/core-js": { + "version": "3.32.1", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.32.1.tgz", + "integrity": "sha512-lqufgNn9NLnESg5mQeYsxQP5w7wrViSj0jr/kv6ECQiByzQkrn1MKvV0L3acttpDqfQrHLwr2KCMgX5b8X+lyQ==", + "hasInstallScript": true, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" + } + }, + "node_modules/core-js-compat": { + "version": "3.32.1", + "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.32.1.tgz", + "integrity": "sha512-GSvKDv4wE0bPnQtjklV101juQ85g6H3rm5PDP20mqlS5j0kXF3pP97YvAu5hl+uFHqMictp3b2VxOHljWMAtuA==", + "dependencies": { + "browserslist": "^4.21.10" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" + } + }, + "node_modules/core-js-pure": { + "version": "3.32.1", + "resolved": "https://registry.npmjs.org/core-js-pure/-/core-js-pure-3.32.1.tgz", + "integrity": "sha512-f52QZwkFVDPf7UEQZGHKx6NYxsxmVGJe5DIvbzOdRMJlmT6yv0KDjR8rmy3ngr/t5wU54c7Sp/qIJH0ppbhVpQ==", + "hasInstallScript": true, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" + } + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==" + }, + "node_modules/cosmiconfig": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.1.0.tgz", + "integrity": "sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==", + "dependencies": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.2.1", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.10.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/crypto-random-string": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", + "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", + "engines": { + "node": ">=8" + } + }, + "node_modules/css-blank-pseudo": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/css-blank-pseudo/-/css-blank-pseudo-3.0.3.tgz", + "integrity": "sha512-VS90XWtsHGqoM0t4KpH053c4ehxZ2E6HtGI7x68YFV0pTo/QmkV/YFA+NnlvK8guxZVNWGQhVNJGC39Q8XF4OQ==", + "dependencies": { + "postcss-selector-parser": "^6.0.9" + }, + "bin": { + "css-blank-pseudo": "dist/cli.cjs" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/css-declaration-sorter": { + "version": "6.4.1", + "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-6.4.1.tgz", + "integrity": "sha512-rtdthzxKuyq6IzqX6jEcIzQF/YqccluefyCYheovBOLhFT/drQA9zj/UbRAa9J7C0o6EG6u3E6g+vKkay7/k3g==", + "engines": { + "node": "^10 || ^12 || >=14" + }, + "peerDependencies": { + "postcss": "^8.0.9" + } + }, + "node_modules/css-has-pseudo": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/css-has-pseudo/-/css-has-pseudo-3.0.4.tgz", + "integrity": "sha512-Vse0xpR1K9MNlp2j5w1pgWIJtm1a8qS0JwS9goFYcImjlHEmywP9VUF05aGBXzGpDJF86QXk4L0ypBmwPhGArw==", + "dependencies": { + "postcss-selector-parser": "^6.0.9" + }, + "bin": { + "css-has-pseudo": "dist/cli.cjs" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/css-loader": { + "version": "6.8.1", + "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-6.8.1.tgz", + "integrity": "sha512-xDAXtEVGlD0gJ07iclwWVkLoZOpEvAWaSyf6W18S2pOC//K8+qUDIx8IIT3D+HjnmkJPQeesOPv5aiUaJsCM2g==", + "dependencies": { + "icss-utils": "^5.1.0", + "postcss": "^8.4.21", + "postcss-modules-extract-imports": "^3.0.0", + "postcss-modules-local-by-default": "^4.0.3", + "postcss-modules-scope": "^3.0.0", + "postcss-modules-values": "^4.0.0", + "postcss-value-parser": "^4.2.0", + "semver": "^7.3.8" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.0.0" + } + }, + "node_modules/css-minimizer-webpack-plugin": { + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/css-minimizer-webpack-plugin/-/css-minimizer-webpack-plugin-3.4.1.tgz", + "integrity": "sha512-1u6D71zeIfgngN2XNRJefc/hY7Ybsxd74Jm4qngIXyUEk7fss3VUzuHxLAq/R8NAba4QU9OUSaMZlbpRc7bM4Q==", + "dependencies": { + "cssnano": "^5.0.6", + "jest-worker": "^27.0.2", + "postcss": "^8.3.5", + "schema-utils": "^4.0.0", + "serialize-javascript": "^6.0.0", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.0.0" + }, + "peerDependenciesMeta": { + "@parcel/css": { + "optional": true + }, + "clean-css": { + "optional": true + }, + "csso": { + "optional": true + }, + "esbuild": { + "optional": true + } + } + }, + "node_modules/css-minimizer-webpack-plugin/node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/css-minimizer-webpack-plugin/node_modules/ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dependencies": { + "fast-deep-equal": "^3.1.3" + }, + "peerDependencies": { + "ajv": "^8.8.2" + } + }, + "node_modules/css-minimizer-webpack-plugin/node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/css-minimizer-webpack-plugin/node_modules/schema-utils": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.2.0.tgz", + "integrity": "sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/css-minimizer-webpack-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-prefers-color-scheme": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/css-prefers-color-scheme/-/css-prefers-color-scheme-6.0.3.tgz", + "integrity": "sha512-4BqMbZksRkJQx2zAjrokiGMd07RqOa2IxIrrN10lyBe9xhn9DEvjUK79J6jkeiv9D9hQFXKb6g1jwU62jziJZA==", + "bin": { + "css-prefers-color-scheme": "dist/cli.cjs" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/css-select": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-4.3.0.tgz", + "integrity": "sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ==", + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^6.0.1", + "domhandler": "^4.3.1", + "domutils": "^2.8.0", + "nth-check": "^2.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/css-select-base-adapter": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/css-select-base-adapter/-/css-select-base-adapter-0.1.1.tgz", + "integrity": "sha512-jQVeeRG70QI08vSTwf1jHxp74JoZsr2XSgETae8/xC8ovSnL2WF87GTLO86Sbwdt2lK4Umg4HnnwMO4YF3Ce7w==" + }, + "node_modules/css-tree": { + "version": "1.0.0-alpha.37", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.0.0-alpha.37.tgz", + "integrity": "sha512-DMxWJg0rnz7UgxKT0Q1HU/L9BeJI0M6ksor0OgqOnF+aRCDWg/N2641HmVyU9KVIu0OVVWOb2IpC9A+BJRnejg==", + "dependencies": { + "mdn-data": "2.0.4", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/css-tree/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-what": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/css-what/-/css-what-6.1.0.tgz", + "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==", + "engines": { + "node": ">= 6" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/css.escape": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/css.escape/-/css.escape-1.5.1.tgz", + "integrity": "sha512-YUifsXXuknHlUsmlgyY0PKzgPOr7/FjCePfHNt0jxm83wHZi44VDMQ7/fGNkjY3/jV1MC+1CmZbaHzugyeRtpg==" + }, + "node_modules/cssdb": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/cssdb/-/cssdb-7.7.1.tgz", + "integrity": "sha512-kM+Fs0BFyhJNeE6wbOrlnRsugRdL6vn7QcON0aBDZ7XRd7RI2pMlk+nxoHuTb4Et+aBobXgK0I+6NGLA0LLgTw==", + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + } + ] + }, + "node_modules/cssesc": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", + "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", + "bin": { + "cssesc": "bin/cssesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cssnano": { + "version": "5.1.15", + "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-5.1.15.tgz", + "integrity": "sha512-j+BKgDcLDQA+eDifLx0EO4XSA56b7uut3BQFH+wbSaSTuGLuiyTa/wbRYthUXX8LC9mLg+WWKe8h+qJuwTAbHw==", + "dependencies": { + "cssnano-preset-default": "^5.2.14", + "lilconfig": "^2.0.3", + "yaml": "^1.10.2" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/cssnano" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/cssnano-preset-default": { + "version": "5.2.14", + "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-5.2.14.tgz", + "integrity": "sha512-t0SFesj/ZV2OTylqQVOrFgEh5uanxbO6ZAdeCrNsUQ6fVuXwYTxJPNAGvGTxHbD68ldIJNec7PyYZDBrfDQ+6A==", + "dependencies": { + "css-declaration-sorter": "^6.3.1", + "cssnano-utils": "^3.1.0", + "postcss-calc": "^8.2.3", + "postcss-colormin": "^5.3.1", + "postcss-convert-values": "^5.1.3", + "postcss-discard-comments": "^5.1.2", + "postcss-discard-duplicates": "^5.1.0", + "postcss-discard-empty": "^5.1.1", + "postcss-discard-overridden": "^5.1.0", + "postcss-merge-longhand": "^5.1.7", + "postcss-merge-rules": "^5.1.4", + "postcss-minify-font-values": "^5.1.0", + "postcss-minify-gradients": "^5.1.1", + "postcss-minify-params": "^5.1.4", + "postcss-minify-selectors": "^5.2.1", + "postcss-normalize-charset": "^5.1.0", + "postcss-normalize-display-values": "^5.1.0", + "postcss-normalize-positions": "^5.1.1", + "postcss-normalize-repeat-style": "^5.1.1", + "postcss-normalize-string": "^5.1.0", + "postcss-normalize-timing-functions": "^5.1.0", + "postcss-normalize-unicode": "^5.1.1", + "postcss-normalize-url": "^5.1.0", + "postcss-normalize-whitespace": "^5.1.1", + "postcss-ordered-values": "^5.1.3", + "postcss-reduce-initial": "^5.1.2", + "postcss-reduce-transforms": "^5.1.0", + "postcss-svgo": "^5.1.0", + "postcss-unique-selectors": "^5.1.1" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/cssnano-utils": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cssnano-utils/-/cssnano-utils-3.1.0.tgz", + "integrity": "sha512-JQNR19/YZhz4psLX/rQ9M83e3z2Wf/HdJbryzte4a3NSuafyp9w/I4U+hx5C2S9g41qlstH7DEWnZaaj83OuEA==", + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/csso": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/csso/-/csso-4.2.0.tgz", + "integrity": "sha512-wvlcdIbf6pwKEk7vHj8/Bkc0B4ylXZruLvOgs9doS5eOsOpuodOV2zJChSpkp+pRpYQLQMeF04nr3Z68Sta9jA==", + "dependencies": { + "css-tree": "^1.1.2" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/csso/node_modules/css-tree": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.1.3.tgz", + "integrity": "sha512-tRpdppF7TRazZrjJ6v3stzv93qxRcSsFmW6cX0Zm2NVKpxE1WV1HblnghVv9TreireHkqI/VDEsfolRF1p6y7Q==", + "dependencies": { + "mdn-data": "2.0.14", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/csso/node_modules/mdn-data": { + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.14.tgz", + "integrity": "sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow==" + }, + "node_modules/csso/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cssom": { + "version": "0.4.4", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.4.4.tgz", + "integrity": "sha512-p3pvU7r1MyyqbTk+WbNJIgJjG2VmTIaB10rI93LzVPrmDJKkzKYMtxxyAvQXR/NS6otuzveI7+7BBq3SjBS2mw==" + }, + "node_modules/cssstyle": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz", + "integrity": "sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A==", + "dependencies": { + "cssom": "~0.3.6" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cssstyle/node_modules/cssom": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", + "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==" + }, + "node_modules/csstype": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.2.tgz", + "integrity": "sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ==" + }, + "node_modules/damerau-levenshtein": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.8.tgz", + "integrity": "sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA==" + }, + "node_modules/data-urls": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", + "integrity": "sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ==", + "dependencies": { + "abab": "^2.0.3", + "whatwg-mimetype": "^2.3.0", + "whatwg-url": "^8.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decimal.js": { + "version": "10.4.3", + "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.4.3.tgz", + "integrity": "sha512-VBBaLc1MgL5XpzgIP7ny5Z6Nx3UrRkIViUkPUdtl9aya5amy3De1gsUUSB1g3+3sExYNjCAsAznmukyxCb1GRA==" + }, + "node_modules/dedent": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-0.7.0.tgz", + "integrity": "sha512-Q6fKUPqnAHAyhiUgFU7BUzLiv0kd8saH9al7tnu5Q/okj6dnupxyTgFIBjVzJATdfIAm9NAsvXNzjaKa+bxVyA==" + }, + "node_modules/deep-equal": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-2.2.2.tgz", + "integrity": "sha512-xjVyBf0w5vH0I42jdAZzOKVldmPgSulmiyPRywoyq7HXC9qdgo17kxJE+rdnif5Tz6+pIrpJI8dCpMNLIGkUiA==", + "dependencies": { + "array-buffer-byte-length": "^1.0.0", + "call-bind": "^1.0.2", + "es-get-iterator": "^1.1.3", + "get-intrinsic": "^1.2.1", + "is-arguments": "^1.1.1", + "is-array-buffer": "^3.0.2", + "is-date-object": "^1.0.5", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "isarray": "^2.0.5", + "object-is": "^1.1.5", + "object-keys": "^1.1.1", + "object.assign": "^4.1.4", + "regexp.prototype.flags": "^1.5.0", + "side-channel": "^1.0.4", + "which-boxed-primitive": "^1.0.2", + "which-collection": "^1.0.1", + "which-typed-array": "^1.1.9" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==" + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/default-gateway": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/default-gateway/-/default-gateway-6.0.3.tgz", + "integrity": "sha512-fwSOJsbbNzZ/CUFpqFBqYfYNLj1NbMPm8MMCIzHjC83iSJRBEGmDUxU+WP661BaBQImeC2yHwXtz+P/O9o+XEg==", + "dependencies": { + "execa": "^5.0.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/define-lazy-prop": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-2.0.0.tgz", + "integrity": "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==", + "engines": { + "node": ">=8" + } + }, + "node_modules/define-properties": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.0.tgz", + "integrity": "sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==", + "dependencies": { + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/destroy": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", + "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/detect-newline": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", + "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", + "engines": { + "node": ">=8" + } + }, + "node_modules/detect-node": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/detect-node/-/detect-node-2.1.0.tgz", + "integrity": "sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==" + }, + "node_modules/detect-port-alt": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/detect-port-alt/-/detect-port-alt-1.1.6.tgz", + "integrity": "sha512-5tQykt+LqfJFBEYaDITx7S7cR7mJ/zQmLXZ2qt5w04ainYZw6tBf9dBunMjVeVOdYVRUzUOE4HkY5J7+uttb5Q==", + "dependencies": { + "address": "^1.0.1", + "debug": "^2.6.0" + }, + "bin": { + "detect": "bin/detect-port", + "detect-port": "bin/detect-port" + }, + "engines": { + "node": ">= 4.2.1" + } + }, + "node_modules/detect-port-alt/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/detect-port-alt/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/didyoumean": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz", + "integrity": "sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==" + }, + "node_modules/diff-sequences": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-27.5.1.tgz", + "integrity": "sha512-k1gCAXAsNgLwEL+Y8Wvl+M6oEFj5bgazfZULpS5CneoPPXRaCCW7dm+q21Ky2VEE5X+VeRDBVg1Pcvvsr4TtNQ==", + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/dlv": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz", + "integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==" + }, + "node_modules/dns-equal": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/dns-equal/-/dns-equal-1.0.0.tgz", + "integrity": "sha512-z+paD6YUQsk+AbGCEM4PrOXSss5gd66QfcVBFTKR/HpFL9jCqikS94HYwKww6fQyO7IxrIIyUu+g0Ka9tUS2Cg==" + }, + "node_modules/dns-packet": { + "version": "5.6.1", + "resolved": "https://registry.npmjs.org/dns-packet/-/dns-packet-5.6.1.tgz", + "integrity": "sha512-l4gcSouhcgIKRvyy99RNVOgxXiicE+2jZoNmaNmZ6JXiGajBOJAesk1OBlJuM5k2c+eudGdLxDqXuPCKIj6kpw==", + "dependencies": { + "@leichtgewicht/ip-codec": "^2.0.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/dom-accessibility-api": { + "version": "0.5.16", + "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz", + "integrity": "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==" + }, + "node_modules/dom-converter": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/dom-converter/-/dom-converter-0.2.0.tgz", + "integrity": "sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA==", + "dependencies": { + "utila": "~0.4" + } + }, + "node_modules/dom-serializer": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.4.1.tgz", + "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", + "entities": "^2.0.0" + }, + "funding": { + "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" + } + }, + "node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ] + }, + "node_modules/domexception": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/domexception/-/domexception-2.0.1.tgz", + "integrity": "sha512-yxJ2mFy/sibVQlu5qHjOkf9J3K6zgmCxgJ94u2EdvDOV09H+32LtRswEcUsmUWN72pVLOEnTSRaIVVzVQgS0dg==", + "dependencies": { + "webidl-conversions": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/domexception/node_modules/webidl-conversions": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-5.0.0.tgz", + "integrity": "sha512-VlZwKPCkYKxQgeSbH5EyngOmRp7Ww7I9rQLERETtf5ofd9pGeswWiOtogpEO850jziPRarreGxn5QIiTqpb2wA==", + "engines": { + "node": ">=8" + } + }, + "node_modules/domhandler": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.3.1.tgz", + "integrity": "sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ==", + "dependencies": { + "domelementtype": "^2.2.0" + }, + "engines": { + "node": ">= 4" + }, + "funding": { + "url": "https://github.com/fb55/domhandler?sponsor=1" + } + }, + "node_modules/domutils": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.8.0.tgz", + "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", + "dependencies": { + "dom-serializer": "^1.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.2.0" + }, + "funding": { + "url": "https://github.com/fb55/domutils?sponsor=1" + } + }, + "node_modules/dot-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-3.0.4.tgz", + "integrity": "sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==", + "dependencies": { + "no-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "node_modules/dotenv": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-10.0.0.tgz", + "integrity": "sha512-rlBi9d8jpv9Sf1klPjNfFAuWDjKLwTIJJ/VxtoTwIR6hnZxcEOQCZg2oIL3MWBYw5GpUDKOEnND7LXTbIpQ03Q==", + "engines": { + "node": ">=10" + } + }, + "node_modules/dotenv-expand": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/dotenv-expand/-/dotenv-expand-5.1.0.tgz", + "integrity": "sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA==" + }, + "node_modules/duplexer": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/duplexer/-/duplexer-0.1.2.tgz", + "integrity": "sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==" + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" + }, + "node_modules/ejs": { + "version": "3.1.9", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.9.tgz", + "integrity": "sha512-rC+QVNMJWv+MtPgkt0y+0rVEIdbtxVADApW9JXrUVlzHetgcyczP/E7DJmWJ4fJCZF2cPcBk0laWO9ZHMG3DmQ==", + "dependencies": { + "jake": "^10.8.5" + }, + "bin": { + "ejs": "bin/cli.js" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/electron-to-chromium": { + "version": "1.4.503", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.503.tgz", + "integrity": "sha512-LF2IQit4B0VrUHFeQkWhZm97KuJSGF2WJqq1InpY+ECpFRkXd8yTIaTtJxsO0OKDmiBYwWqcrNaXOurn2T2wiA==" + }, + "node_modules/emittery": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.8.1.tgz", + "integrity": "sha512-uDfvUjVrfGJJhymx/kz6prltenw1u7WrCg1oa94zYY8xxVpLLUu045LAT0dhDZdXG58/EpPL/5kA180fQ/qudg==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "node_modules/emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==" + }, + "node_modules/emojis-list": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-3.0.0.tgz", + "integrity": "sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==", + "engines": { + "node": ">= 4" + } + }, + "node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/enhanced-resolve": { + "version": "5.15.0", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.15.0.tgz", + "integrity": "sha512-LXYT42KJ7lpIKECr2mAXIaMldcNCh/7E0KBKOu4KSfkHmP+mZmSs+8V5gBAqisWBy0OO4W5Oyys0GO1Y8KtdKg==", + "dependencies": { + "graceful-fs": "^4.2.4", + "tapable": "^2.2.0" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/entities": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", + "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==", + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/error-stack-parser": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/error-stack-parser/-/error-stack-parser-2.1.4.tgz", + "integrity": "sha512-Sk5V6wVazPhq5MhpO+AUxJn5x7XSXGl1R93Vn7i+zS15KDVxQijejNCrz8340/2bgLBjR9GtEG8ZVKONDjcqGQ==", + "dependencies": { + "stackframe": "^1.3.4" + } + }, + "node_modules/es-abstract": { + "version": "1.22.1", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.22.1.tgz", + "integrity": "sha512-ioRRcXMO6OFyRpyzV3kE1IIBd4WG5/kltnzdxSCqoP8CMGs/Li+M1uF5o7lOkZVFjDs+NLesthnF66Pg/0q0Lw==", + "dependencies": { + "array-buffer-byte-length": "^1.0.0", + "arraybuffer.prototype.slice": "^1.0.1", + "available-typed-arrays": "^1.0.5", + "call-bind": "^1.0.2", + "es-set-tostringtag": "^2.0.1", + "es-to-primitive": "^1.2.1", + "function.prototype.name": "^1.1.5", + "get-intrinsic": "^1.2.1", + "get-symbol-description": "^1.0.0", + "globalthis": "^1.0.3", + "gopd": "^1.0.1", + "has": "^1.0.3", + "has-property-descriptors": "^1.0.0", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.5", + "is-array-buffer": "^3.0.2", + "is-callable": "^1.2.7", + "is-negative-zero": "^2.0.2", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "is-string": "^1.0.7", + "is-typed-array": "^1.1.10", + "is-weakref": "^1.0.2", + "object-inspect": "^1.12.3", + "object-keys": "^1.1.1", + "object.assign": "^4.1.4", + "regexp.prototype.flags": "^1.5.0", + "safe-array-concat": "^1.0.0", + "safe-regex-test": "^1.0.0", + "string.prototype.trim": "^1.2.7", + "string.prototype.trimend": "^1.0.6", + "string.prototype.trimstart": "^1.0.6", + "typed-array-buffer": "^1.0.0", + "typed-array-byte-length": "^1.0.0", + "typed-array-byte-offset": "^1.0.0", + "typed-array-length": "^1.0.4", + "unbox-primitive": "^1.0.2", + "which-typed-array": "^1.1.10" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/es-array-method-boxes-properly": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-array-method-boxes-properly/-/es-array-method-boxes-properly-1.0.0.tgz", + "integrity": "sha512-wd6JXUmyHmt8T5a2xreUwKcGPq6f1f+WwIJkijUqiGcJz1qqnZgP6XIK+QyIWU5lT7imeNxUll48bziG+TSYcA==" + }, + "node_modules/es-get-iterator": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/es-get-iterator/-/es-get-iterator-1.1.3.tgz", + "integrity": "sha512-sPZmqHBe6JIiTfN5q2pEi//TwxmAFHwj/XEuYjTuse78i8KxaqMTTzxPoFKuzRpDpTJ+0NAbpfenkmH2rePtuw==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.3", + "has-symbols": "^1.0.3", + "is-arguments": "^1.1.1", + "is-map": "^2.0.2", + "is-set": "^2.0.2", + "is-string": "^1.0.7", + "isarray": "^2.0.5", + "stop-iteration-iterator": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/es-iterator-helpers": { + "version": "1.0.13", + "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.0.13.tgz", + "integrity": "sha512-LK3VGwzvaPWobO8xzXXGRUOGw8Dcjyfk62CsY/wfHN75CwsJPbuypOYJxK6g5RyEL8YDjIWcl6jgd8foO6mmrA==", + "dependencies": { + "asynciterator.prototype": "^1.0.0", + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.21.3", + "es-set-tostringtag": "^2.0.1", + "function-bind": "^1.1.1", + "get-intrinsic": "^1.2.1", + "globalthis": "^1.0.3", + "has-property-descriptors": "^1.0.0", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.5", + "iterator.prototype": "^1.1.0", + "safe-array-concat": "^1.0.0" + } + }, + "node_modules/es-module-lexer": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.3.0.tgz", + "integrity": "sha512-vZK7T0N2CBmBOixhmjdqx2gWVbFZ4DXZ/NyRMZVlJXPa7CyFS+/a4QQsDGDQy9ZfEzxFuNEsMLeQJnKP2p5/JA==" + }, + "node_modules/es-set-tostringtag": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz", + "integrity": "sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==", + "dependencies": { + "get-intrinsic": "^1.1.3", + "has": "^1.0.3", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-shim-unscopables": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz", + "integrity": "sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w==", + "dependencies": { + "has": "^1.0.3" + } + }, + "node_modules/es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dependencies": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" + }, + "node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/escodegen": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-2.1.0.tgz", + "integrity": "sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w==", + "dependencies": { + "esprima": "^4.0.1", + "estraverse": "^5.2.0", + "esutils": "^2.0.2" + }, + "bin": { + "escodegen": "bin/escodegen.js", + "esgenerate": "bin/esgenerate.js" + }, + "engines": { + "node": ">=6.0" + }, + "optionalDependencies": { + "source-map": "~0.6.1" + } + }, + "node_modules/escodegen/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/eslint": { + "version": "8.48.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.48.0.tgz", + "integrity": "sha512-sb6DLeIuRXxeM1YljSe1KEx9/YYeZFQWcV8Rq9HfigmdDEugjLEVEa1ozDjL6YDjBpQHPJxJzze+alxi4T3OLg==", + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.6.1", + "@eslint/eslintrc": "^2.1.2", + "@eslint/js": "8.48.0", + "@humanwhocodes/config-array": "^0.11.10", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.2.2", + "eslint-visitor-keys": "^3.4.3", + "espree": "^9.6.1", + "esquery": "^1.4.2", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "globals": "^13.19.0", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "is-path-inside": "^3.0.3", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-config-react-app": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/eslint-config-react-app/-/eslint-config-react-app-7.0.1.tgz", + "integrity": "sha512-K6rNzvkIeHaTd8m/QEh1Zko0KI7BACWkkneSs6s9cKZC/J27X3eZR6Upt1jkmZ/4FK+XUOPPxMEN7+lbUXfSlA==", + "dependencies": { + "@babel/core": "^7.16.0", + "@babel/eslint-parser": "^7.16.3", + "@rushstack/eslint-patch": "^1.1.0", + "@typescript-eslint/eslint-plugin": "^5.5.0", + "@typescript-eslint/parser": "^5.5.0", + "babel-preset-react-app": "^10.0.1", + "confusing-browser-globals": "^1.0.11", + "eslint-plugin-flowtype": "^8.0.3", + "eslint-plugin-import": "^2.25.3", + "eslint-plugin-jest": "^25.3.0", + "eslint-plugin-jsx-a11y": "^6.5.1", + "eslint-plugin-react": "^7.27.1", + "eslint-plugin-react-hooks": "^4.3.0", + "eslint-plugin-testing-library": "^5.0.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "eslint": "^8.0.0" + } + }, + "node_modules/eslint-import-resolver-node": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz", + "integrity": "sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==", + "dependencies": { + "debug": "^3.2.7", + "is-core-module": "^2.13.0", + "resolve": "^1.22.4" + } + }, + "node_modules/eslint-import-resolver-node/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/eslint-module-utils": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.8.0.tgz", + "integrity": "sha512-aWajIYfsqCKRDgUfjEXNN/JlrzauMuSEy5sbd7WXbtW3EH6A6MpwEh42c7qD+MqQo9QMJ6fWLAeIJynx0g6OAw==", + "dependencies": { + "debug": "^3.2.7" + }, + "engines": { + "node": ">=4" + }, + "peerDependenciesMeta": { + "eslint": { + "optional": true + } + } + }, + "node_modules/eslint-module-utils/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/eslint-plugin-flowtype": { + "version": "8.0.3", + "resolved": "https://registry.npmjs.org/eslint-plugin-flowtype/-/eslint-plugin-flowtype-8.0.3.tgz", + "integrity": "sha512-dX8l6qUL6O+fYPtpNRideCFSpmWOUVx5QcaGLVqe/vlDiBSe4vYljDWDETwnyFzpl7By/WVIu6rcrniCgH9BqQ==", + "dependencies": { + "lodash": "^4.17.21", + "string-natural-compare": "^3.0.1" + }, + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "@babel/plugin-syntax-flow": "^7.14.5", + "@babel/plugin-transform-react-jsx": "^7.14.9", + "eslint": "^8.1.0" + } + }, + "node_modules/eslint-plugin-import": { + "version": "2.28.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.28.1.tgz", + "integrity": "sha512-9I9hFlITvOV55alzoKBI+K9q74kv0iKMeY6av5+umsNwayt59fz692daGyjR+oStBQgx6nwR9rXldDev3Clw+A==", + "dependencies": { + "array-includes": "^3.1.6", + "array.prototype.findlastindex": "^1.2.2", + "array.prototype.flat": "^1.3.1", + "array.prototype.flatmap": "^1.3.1", + "debug": "^3.2.7", + "doctrine": "^2.1.0", + "eslint-import-resolver-node": "^0.3.7", + "eslint-module-utils": "^2.8.0", + "has": "^1.0.3", + "is-core-module": "^2.13.0", + "is-glob": "^4.0.3", + "minimatch": "^3.1.2", + "object.fromentries": "^2.0.6", + "object.groupby": "^1.0.0", + "object.values": "^1.1.6", + "semver": "^6.3.1", + "tsconfig-paths": "^3.14.2" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8" + } + }, + "node_modules/eslint-plugin-import/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/eslint-plugin-import/node_modules/doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/eslint-plugin-import/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint-plugin-jest": { + "version": "25.7.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-jest/-/eslint-plugin-jest-25.7.0.tgz", + "integrity": "sha512-PWLUEXeeF7C9QGKqvdSbzLOiLTx+bno7/HC9eefePfEb257QFHg7ye3dh80AZVkaa/RQsBB1Q/ORQvg2X7F0NQ==", + "dependencies": { + "@typescript-eslint/experimental-utils": "^5.0.0" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || >=16.0.0" + }, + "peerDependencies": { + "@typescript-eslint/eslint-plugin": "^4.0.0 || ^5.0.0", + "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "@typescript-eslint/eslint-plugin": { + "optional": true + }, + "jest": { + "optional": true + } + } + }, + "node_modules/eslint-plugin-jsx-a11y": { + "version": "6.7.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-jsx-a11y/-/eslint-plugin-jsx-a11y-6.7.1.tgz", + "integrity": "sha512-63Bog4iIethyo8smBklORknVjB0T2dwB8Mr/hIC+fBS0uyHdYYpzM/Ed+YC8VxTjlXHEWFOdmgwcDn1U2L9VCA==", + "dependencies": { + "@babel/runtime": "^7.20.7", + "aria-query": "^5.1.3", + "array-includes": "^3.1.6", + "array.prototype.flatmap": "^1.3.1", + "ast-types-flow": "^0.0.7", + "axe-core": "^4.6.2", + "axobject-query": "^3.1.1", + "damerau-levenshtein": "^1.0.8", + "emoji-regex": "^9.2.2", + "has": "^1.0.3", + "jsx-ast-utils": "^3.3.3", + "language-tags": "=1.0.5", + "minimatch": "^3.1.2", + "object.entries": "^1.1.6", + "object.fromentries": "^2.0.6", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=4.0" + }, + "peerDependencies": { + "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8" + } + }, + "node_modules/eslint-plugin-jsx-a11y/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint-plugin-react": { + "version": "7.33.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.33.2.tgz", + "integrity": "sha512-73QQMKALArI8/7xGLNI/3LylrEYrlKZSb5C9+q3OtOewTnMQi5cT+aE9E41sLCmli3I9PGGmD1yiZydyo4FEPw==", + "dependencies": { + "array-includes": "^3.1.6", + "array.prototype.flatmap": "^1.3.1", + "array.prototype.tosorted": "^1.1.1", + "doctrine": "^2.1.0", + "es-iterator-helpers": "^1.0.12", + "estraverse": "^5.3.0", + "jsx-ast-utils": "^2.4.1 || ^3.0.0", + "minimatch": "^3.1.2", + "object.entries": "^1.1.6", + "object.fromentries": "^2.0.6", + "object.hasown": "^1.1.2", + "object.values": "^1.1.6", + "prop-types": "^15.8.1", + "resolve": "^2.0.0-next.4", + "semver": "^6.3.1", + "string.prototype.matchall": "^4.0.8" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8" + } + }, + "node_modules/eslint-plugin-react-hooks": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.6.0.tgz", + "integrity": "sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==", + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/eslint-plugin-react/node_modules/doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/eslint-plugin-react/node_modules/resolve": { + "version": "2.0.0-next.4", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-2.0.0-next.4.tgz", + "integrity": "sha512-iMDbmAWtfU+MHpxt/I5iWI7cY6YVEZUQ3MBgPQ++XD1PELuJHIl82xBmObyP2KyQmkNB2dsqF7seoQQiAn5yDQ==", + "dependencies": { + "is-core-module": "^2.9.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/eslint-plugin-react/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint-plugin-testing-library": { + "version": "5.11.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-testing-library/-/eslint-plugin-testing-library-5.11.1.tgz", + "integrity": "sha512-5eX9e1Kc2PqVRed3taaLnAAqPZGEX75C+M/rXzUAI3wIg/ZxzUm1OVAwfe/O+vE+6YXOLetSe9g5GKD2ecXipw==", + "dependencies": { + "@typescript-eslint/utils": "^5.58.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0", + "npm": ">=6" + }, + "peerDependencies": { + "eslint": "^7.5.0 || ^8.0.0" + } + }, + "node_modules/eslint-scope": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", + "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "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==", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-webpack-plugin": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/eslint-webpack-plugin/-/eslint-webpack-plugin-3.2.0.tgz", + "integrity": "sha512-avrKcGncpPbPSUHX6B3stNGzkKFto3eL+DKM4+VyMrVnhPc3vRczVlCq3uhuFOdRvDHTVXuzwk1ZKUrqDQHQ9w==", + "dependencies": { + "@types/eslint": "^7.29.0 || ^8.4.1", + "jest-worker": "^28.0.2", + "micromatch": "^4.0.5", + "normalize-path": "^3.0.0", + "schema-utils": "^4.0.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "eslint": "^7.0.0 || ^8.0.0", + "webpack": "^5.0.0" + } + }, + "node_modules/eslint-webpack-plugin/node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/eslint-webpack-plugin/node_modules/ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dependencies": { + "fast-deep-equal": "^3.1.3" + }, + "peerDependencies": { + "ajv": "^8.8.2" + } + }, + "node_modules/eslint-webpack-plugin/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint-webpack-plugin/node_modules/jest-worker": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-28.1.3.tgz", + "integrity": "sha512-CqRA220YV/6jCo8VWvAt1KKx6eek1VIHMPeLEbpcfSfkEeWyBNppynM/o6q+Wmw+sOhos2ml34wZbSX3G13//g==", + "dependencies": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/eslint-webpack-plugin/node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/eslint-webpack-plugin/node_modules/schema-utils": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.2.0.tgz", + "integrity": "sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/eslint-webpack-plugin/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/eslint/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/eslint/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==" + }, + "node_modules/eslint/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/eslint/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/eslint/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==" + }, + "node_modules/eslint/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==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint/node_modules/globals": { + "version": "13.21.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.21.0.tgz", + "integrity": "sha512-ybyme3s4yy/t/3s35bewwXKOf7cvzfreG2lH0lZl0JB7I4GxRP2ghxOK/Nb9EkRXdbBXZLfq/p/0W2JUONB/Gg==", + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/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==", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/eslint/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/esquery": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", + "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "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==", + "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==", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estree-walker": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-1.0.1.tgz", + "integrity": "sha512-1fMXF3YP4pZZVozF8j/ZLfvnR8NSIljt56UhbZ5PeeDmmGHpgpdwQt7ITlGvYaQukCvuBRMLEiKiYC+oeIg4cg==" + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/eventemitter3": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", + "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==" + }, + "node_modules/events": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "engines": { + "node": ">=0.8.x" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/expect/-/expect-27.5.1.tgz", + "integrity": "sha512-E1q5hSUG2AmYQwQJ041nvgpkODHQvB+RKlB4IYdru6uJsyFTRyZAP463M+1lINorwbqAmUggi6+WwkD8lCS/Dw==", + "dependencies": { + "@jest/types": "^27.5.1", + "jest-get-type": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/express": { + "version": "4.18.2", + "resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz", + "integrity": "sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==", + "dependencies": { + "accepts": "~1.3.8", + "array-flatten": "1.1.1", + "body-parser": "1.20.1", + "content-disposition": "0.5.4", + "content-type": "~1.0.4", + "cookie": "0.5.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "2.0.0", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "1.2.0", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.7", + "qs": "6.11.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.2.1", + "send": "0.18.0", + "serve-static": "1.15.0", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/express/node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" + }, + "node_modules/express/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/express/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "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==" + }, + "node_modules/fast-glob": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.1.tgz", + "integrity": "sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==", + "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==", + "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==" + }, + "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==" + }, + "node_modules/fastq": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz", + "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==", + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/faye-websocket": { + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.4.tgz", + "integrity": "sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==", + "dependencies": { + "websocket-driver": ">=0.5.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/fb-watchman": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz", + "integrity": "sha512-p5161BqbuCaSnB8jIbzQHOlpgsPmK5rJVDfDKO91Axs5NC1uu3HRQm6wt9cd9/+GtQQIO53JdGXXoyDpTAsgYA==", + "dependencies": { + "bser": "2.1.1" + } + }, + "node_modules/file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dependencies": { + "flat-cache": "^3.0.4" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/file-loader": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/file-loader/-/file-loader-6.2.0.tgz", + "integrity": "sha512-qo3glqyTa61Ytg4u73GultjHGjdRyig3tG6lPtyX/jOEJvHif9uB0/OCI2Kif6ctF3caQTW2G5gym21oAsI4pw==", + "dependencies": { + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/filelist": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.4.tgz", + "integrity": "sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==", + "dependencies": { + "minimatch": "^5.0.1" + } + }, + "node_modules/filelist/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==", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/filelist/node_modules/minimatch": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", + "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/filesize": { + "version": "8.0.7", + "resolved": "https://registry.npmjs.org/filesize/-/filesize-8.0.7.tgz", + "integrity": "sha512-pjmC+bkIF8XI7fWaH8KxHcZL3DPybs1roSKP4rKDvy20tAWwIObE4+JIseG2byfGKhud5ZnM4YSGKBz7Sh0ndQ==", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/finalhandler": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz", + "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==", + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "statuses": "2.0.1", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/finalhandler/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/finalhandler/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/find-cache-dir": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.2.tgz", + "integrity": "sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==", + "dependencies": { + "commondir": "^1.0.1", + "make-dir": "^3.0.2", + "pkg-dir": "^4.1.0" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/avajs/find-cache-dir?sponsor=1" + } + }, + "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==", + "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": "3.1.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.1.0.tgz", + "integrity": "sha512-OHx4Qwrrt0E4jEIcI5/Xb+f+QmJYNj2rrK8wiIdQOIrB9WrrJL8cjZvXdXuBTkkEwEqLycb5BeZDV1o2i9bTew==", + "dependencies": { + "flatted": "^3.2.7", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + }, + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/flatted": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz", + "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==" + }, + "node_modules/follow-redirects": { + "version": "1.15.5", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.5.tgz", + "integrity": "sha512-vSFWUON1B+yAw1VN4xMfxgn5fTUiaOzAJCKBwIIgT/+7CuGy9+r+5gITvP62j3RmaD5Ph65UaERdOSRGUzZtgw==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/for-each": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", + "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", + "dependencies": { + "is-callable": "^1.1.3" + } + }, + "node_modules/fork-ts-checker-webpack-plugin": { + "version": "6.5.3", + "resolved": "https://registry.npmjs.org/fork-ts-checker-webpack-plugin/-/fork-ts-checker-webpack-plugin-6.5.3.tgz", + "integrity": "sha512-SbH/l9ikmMWycd5puHJKTkZJKddF4iRLyW3DeZ08HTI7NGyLS38MXd/KGgeWumQO7YNQbW2u/NtPT2YowbPaGQ==", + "dependencies": { + "@babel/code-frame": "^7.8.3", + "@types/json-schema": "^7.0.5", + "chalk": "^4.1.0", + "chokidar": "^3.4.2", + "cosmiconfig": "^6.0.0", + "deepmerge": "^4.2.2", + "fs-extra": "^9.0.0", + "glob": "^7.1.6", + "memfs": "^3.1.2", + "minimatch": "^3.0.4", + "schema-utils": "2.7.0", + "semver": "^7.3.2", + "tapable": "^1.0.0" + }, + "engines": { + "node": ">=10", + "yarn": ">=1.0.0" + }, + "peerDependencies": { + "eslint": ">= 6", + "typescript": ">= 2.7", + "vue-template-compiler": "*", + "webpack": ">= 4" + }, + "peerDependenciesMeta": { + "eslint": { + "optional": true + }, + "vue-template-compiler": { + "optional": true + } + } + }, + "node_modules/fork-ts-checker-webpack-plugin/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/fork-ts-checker-webpack-plugin/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/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==" + }, + "node_modules/fork-ts-checker-webpack-plugin/node_modules/cosmiconfig": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-6.0.0.tgz", + "integrity": "sha512-xb3ZL6+L8b9JLLCx3ZdoZy4+2ECphCMo2PwqgP1tlfVq6M6YReyzBJtvWWtbDSpNr9hn96pkCiZqUcFEc+54Qg==", + "dependencies": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.1.0", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.7.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/node_modules/schema-utils": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-2.7.0.tgz", + "integrity": "sha512-0ilKFI6QQF5nxDZLFn2dMjvc4hjg/Wkg7rHd3jK6/A4a1Hl9VFdQWvgB1UMGoU94pad1P/8N7fMcEnLnSiju8A==", + "dependencies": { + "@types/json-schema": "^7.0.4", + "ajv": "^6.12.2", + "ajv-keywords": "^3.4.1" + }, + "engines": { + "node": ">= 8.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/node_modules/tapable": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-1.1.3.tgz", + "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/form-data": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", + "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fraction.js": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.2.1.tgz", + "integrity": "sha512-/KxoyCnPM0GwYI4NN0Iag38Tqt+od3/mLuguepLgCAKPn0ZhC544nssAW0tG2/00zXEYl9W+7hwAIpLHo6Oc7Q==", + "engines": { + "node": "*" + }, + "funding": { + "type": "patreon", + "url": "https://www.patreon.com/infusion" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fs-extra": { + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.1.0.tgz", + "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==", + "dependencies": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/fs-monkey": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/fs-monkey/-/fs-monkey-1.0.4.tgz", + "integrity": "sha512-INM/fWAxMICjttnD0DX1rBvinKskj5G1w+oy/pnm9u/tSlnBrzFonJMcalKJ30P8RRsPzKcCG7Q8l0jx5Fh9YQ==" + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + }, + "node_modules/function.prototype.name": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz", + "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.0", + "functions-have-names": "^1.2.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/functions-have-names": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", + "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-intrinsic": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.1.tgz", + "integrity": "sha512-2DcsyfABl+gVHEfCOaTrWgyt+tb6MSEGmKq+kI5HwLbIYgjgmMcV8KQ41uaKz1xxUcn9tJtgFbQUEVcEbd0FYw==", + "dependencies": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-own-enumerable-property-symbols": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/get-own-enumerable-property-symbols/-/get-own-enumerable-property-symbols-3.0.2.tgz", + "integrity": "sha512-I0UBV/XOz1XkIJHEUDMZAbzCThU/H8DxmSfmdGcKPnVhu2VfFqr34jr9777IyaTYvxjedWhqVIilEDsCdP5G6g==" + }, + "node_modules/get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/get-symbol-description": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz", + "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "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==", + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/glob-to-regexp": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz", + "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==" + }, + "node_modules/global-modules": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-2.0.0.tgz", + "integrity": "sha512-NGbfmJBp9x8IxyJSd1P+otYK8vonoJactOogrVfFRIAEY1ukil8RSKDz2Yo7wh1oihl51l/r6W4epkeKJHqL8A==", + "dependencies": { + "global-prefix": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/global-prefix": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-3.0.0.tgz", + "integrity": "sha512-awConJSVCHVGND6x3tmMaKcQvwXLhjdkmomy2W+Goaui8YPgYgXJZewhg3fWC+DlfqqQuWg8AwqjGTD2nAPVWg==", + "dependencies": { + "ini": "^1.3.5", + "kind-of": "^6.0.2", + "which": "^1.3.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/global-prefix/node_modules/which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "which": "bin/which" + } + }, + "node_modules/globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "engines": { + "node": ">=4" + } + }, + "node_modules/globalthis": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.3.tgz", + "integrity": "sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==", + "dependencies": { + "define-properties": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/globby": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dependencies": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/gopd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", + "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", + "dependencies": { + "get-intrinsic": "^1.1.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==" + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==" + }, + "node_modules/gzip-size": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/gzip-size/-/gzip-size-6.0.0.tgz", + "integrity": "sha512-ax7ZYomf6jqPTQ4+XCpUGyXKHk5WweS+e05MBO4/y3WJ5RkmPXNKvX+bx1behVILVwr6JSQvZAku021CHPXG3Q==", + "dependencies": { + "duplexer": "^0.1.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/handle-thing": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/handle-thing/-/handle-thing-2.0.1.tgz", + "integrity": "sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg==" + }, + "node_modules/harmony-reflect": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/harmony-reflect/-/harmony-reflect-1.6.2.tgz", + "integrity": "sha512-HIp/n38R9kQjDEziXyDTuW3vvoxxyxjxFzXLrBr18uB47GnSt+G9D29fqrpM5ZkspMcPICud3XsBJQ4Y2URg8g==" + }, + "node_modules/has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dependencies": { + "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/has-bigints": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", + "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "engines": { + "node": ">=4" + } + }, + "node_modules/has-property-descriptors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz", + "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==", + "dependencies": { + "get-intrinsic": "^1.1.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz", + "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-tostringtag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz", + "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", + "dependencies": { + "has-symbols": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/he": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", + "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", + "bin": { + "he": "bin/he" + } + }, + "node_modules/hoopy": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/hoopy/-/hoopy-0.1.4.tgz", + "integrity": "sha512-HRcs+2mr52W0K+x8RzcLzuPPmVIKMSv97RGHy0Ea9y/mpcaK+xTrjICA04KAHi4GRzxliNqNJEFYWHghy3rSfQ==", + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/hpack.js": { + "version": "2.1.6", + "resolved": "https://registry.npmjs.org/hpack.js/-/hpack.js-2.1.6.tgz", + "integrity": "sha512-zJxVehUdMGIKsRaNt7apO2Gqp0BdqW5yaiGHXXmbpvxgBYVZnAql+BJb4RO5ad2MgpbZKn5G6nMnegrH1FcNYQ==", + "dependencies": { + "inherits": "^2.0.1", + "obuf": "^1.0.0", + "readable-stream": "^2.0.1", + "wbuf": "^1.1.0" + } + }, + "node_modules/hpack.js/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==" + }, + "node_modules/hpack.js/node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/hpack.js/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/hpack.js/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/html-encoding-sniffer": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-2.0.1.tgz", + "integrity": "sha512-D5JbOMBIR/TVZkubHT+OyT2705QvogUW4IBn6nHd756OwieSF9aDYFj4dv6HHEVGYbHaLETa3WggZYWWMyy3ZQ==", + "dependencies": { + "whatwg-encoding": "^1.0.5" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/html-entities": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-2.4.0.tgz", + "integrity": "sha512-igBTJcNNNhvZFRtm8uA6xMY6xYleeDwn3PeBCkDz7tHttv4F2hsDI2aPgNERWzvRcNYHNT3ymRaQzllmXj4YsQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/mdevils" + }, + { + "type": "patreon", + "url": "https://patreon.com/mdevils" + } + ] + }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==" + }, + "node_modules/html-minifier-terser": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/html-minifier-terser/-/html-minifier-terser-6.1.0.tgz", + "integrity": "sha512-YXxSlJBZTP7RS3tWnQw74ooKa6L9b9i9QYXY21eUEvhZ3u9XLfv6OnFsQq6RxkhHygsaUMvYsZRV5rU/OVNZxw==", + "dependencies": { + "camel-case": "^4.1.2", + "clean-css": "^5.2.2", + "commander": "^8.3.0", + "he": "^1.2.0", + "param-case": "^3.0.4", + "relateurl": "^0.2.7", + "terser": "^5.10.0" + }, + "bin": { + "html-minifier-terser": "cli.js" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/html-webpack-plugin": { + "version": "5.5.3", + "resolved": "https://registry.npmjs.org/html-webpack-plugin/-/html-webpack-plugin-5.5.3.tgz", + "integrity": "sha512-6YrDKTuqaP/TquFH7h4srYWsZx+x6k6+FbsTm0ziCwGHDP78Unr1r9F/H4+sGmMbX08GQcJ+K64x55b+7VM/jg==", + "dependencies": { + "@types/html-minifier-terser": "^6.0.0", + "html-minifier-terser": "^6.0.2", + "lodash": "^4.17.21", + "pretty-error": "^4.0.0", + "tapable": "^2.0.0" + }, + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/html-webpack-plugin" + }, + "peerDependencies": { + "webpack": "^5.20.0" + } + }, + "node_modules/htmlparser2": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-6.1.0.tgz", + "integrity": "sha512-gyyPk6rgonLFEDGoeRgQNaEUvdJ4ktTmmUh/h2t7s+M8oPpIPxgNACWa+6ESR57kXstwqPiCut0V8NRpcwgU7A==", + "funding": [ + "https://github.com/fb55/htmlparser2?sponsor=1", + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ], + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.0.0", + "domutils": "^2.5.2", + "entities": "^2.0.0" + } + }, + "node_modules/http-deceiver": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/http-deceiver/-/http-deceiver-1.2.7.tgz", + "integrity": "sha512-LmpOGxTfbpgtGVxJrj5k7asXHCgNZp5nLfp+hWc8QQRqtb7fUy6kRY3BO1h9ddF6yIPYUARgxGOwB42DnxIaNw==" + }, + "node_modules/http-errors": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", + "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", + "dependencies": { + "depd": "2.0.0", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "toidentifier": "1.0.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/http-parser-js": { + "version": "0.5.8", + "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.8.tgz", + "integrity": "sha512-SGeBX54F94Wgu5RH3X5jsDtf4eHyRogWX1XGT3b4HuW3tQPM4AaBzoUji/4AAJNXCEOWZ5O0DgZmJw1947gD5Q==" + }, + "node_modules/http-proxy": { + "version": "1.18.1", + "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.1.tgz", + "integrity": "sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==", + "dependencies": { + "eventemitter3": "^4.0.0", + "follow-redirects": "^1.0.0", + "requires-port": "^1.0.0" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/http-proxy-agent": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", + "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", + "dependencies": { + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/http-proxy-middleware": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-2.0.6.tgz", + "integrity": "sha512-ya/UeJ6HVBYxrgYotAZo1KvPWlgB48kUJLDePFeneHsVujFaW5WNj2NgWCAE//B1Dl02BIfYlpNgBy8Kf8Rjmw==", + "dependencies": { + "@types/http-proxy": "^1.17.8", + "http-proxy": "^1.18.1", + "is-glob": "^4.0.1", + "is-plain-obj": "^3.0.0", + "micromatch": "^4.0.2" + }, + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "@types/express": "^4.17.13" + }, + "peerDependenciesMeta": { + "@types/express": { + "optional": true + } + } + }, + "node_modules/https-proxy-agent": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", + "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", + "dependencies": { + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/icss-utils": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-5.1.0.tgz", + "integrity": "sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==", + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/idb": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/idb/-/idb-7.1.1.tgz", + "integrity": "sha512-gchesWBzyvGHRO9W8tzUWFDycow5gwjvFKfyV9FF32Y7F50yZMp7mP+T2mJIWFx49zicqyC4uefHM17o6xKIVQ==" + }, + "node_modules/identity-obj-proxy": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/identity-obj-proxy/-/identity-obj-proxy-3.0.0.tgz", + "integrity": "sha512-00n6YnVHKrinT9t0d9+5yZC6UBNJANpYEQvL2LlX6Ab9lnmxzIRcEmTPuyGScvl1+jKuCICX1Z0Ab1pPKKdikA==", + "dependencies": { + "harmony-reflect": "^1.4.6" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/ignore": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz", + "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==", + "engines": { + "node": ">= 4" + } + }, + "node_modules/immer": { + "version": "9.0.21", + "resolved": "https://registry.npmjs.org/immer/-/immer-9.0.21.tgz", + "integrity": "sha512-bc4NBHqOqSfRW7POMkHd51LvClaeMXpm8dx0e8oE2GORbq5aRK7Bxl4FyzVLdGtLmvLKL7BTDBG5ACQm4HWjTA==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/immer" + } + }, + "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==", + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/import-fresh/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==", + "engines": { + "node": ">=4" + } + }, + "node_modules/import-local": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.1.0.tgz", + "integrity": "sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==", + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + }, + "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==", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/indent-string": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", + "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/ini": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==" + }, + "node_modules/internal-slot": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.5.tgz", + "integrity": "sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==", + "dependencies": { + "get-intrinsic": "^1.2.0", + "has": "^1.0.3", + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/ipaddr.js": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-2.1.0.tgz", + "integrity": "sha512-LlbxQ7xKzfBusov6UMi4MFpEg0m+mAm9xyNGEduwXMEDuf4WfzB/RZwMVYEd7IKGvh4IUkEXYxtAVu9T3OelJQ==", + "engines": { + "node": ">= 10" + } + }, + "node_modules/is-arguments": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz", + "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==", + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-array-buffer": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.2.tgz", + "integrity": "sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.2.0", + "is-typed-array": "^1.1.10" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==" + }, + "node_modules/is-async-function": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.0.0.tgz", + "integrity": "sha512-Y1JXKrfykRJGdlDwdKlLpLyMIiWqWvuSd17TvZk68PLAOGOoF4Xyav1z0Xhoi+gCYjZVeC5SI+hYFOfvXmGRCA==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-bigint": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", + "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", + "dependencies": { + "has-bigints": "^1.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-boolean-object": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz", + "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==", + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-callable": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", + "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-core-module": { + "version": "2.13.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.13.0.tgz", + "integrity": "sha512-Z7dk6Qo8pOCp3l4tsX2C5ZVas4V+UxwQodwZhLopL91TX8UyyHEXafPcyoeeWuLrwzHcr3igO78wNLwHJHsMCQ==", + "dependencies": { + "has": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-date-object": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", + "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-docker": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", + "bin": { + "is-docker": "cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-finalizationregistry": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.0.2.tgz", + "integrity": "sha512-0by5vtUJs8iFQb5TYUHHPudOR+qXYIMKtiUzvLIZITZUjknFmziyBJuLhVRc+Ds0dREFlskDNJKYIdIzu/9pfw==", + "dependencies": { + "call-bind": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/is-generator-function": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz", + "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-map": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.2.tgz", + "integrity": "sha512-cOZFQQozTha1f4MxLFzlgKYPTyj26picdZTx82hbc/Xf4K/tZOOXSCkMvU4pKioRXGDLJRn0GM7Upe7kR721yg==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-module": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz", + "integrity": "sha512-51ypPSPCoTEIN9dy5Oy+h4pShgJmPCygKfyRCISBI+JoWT/2oJvK8QPxmwv7b/p239jXrm9M1mlQbyKJ5A152g==" + }, + "node_modules/is-negative-zero": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", + "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-number-object": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.7.tgz", + "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz", + "integrity": "sha512-l4RyHgRqGN4Y3+9JHVrNqO+tN0rV5My76uW5/nuO4K1b6vw5G8d/cmFjP9tRfEsdhZNt0IFdZuK/c2Vr4Nb+Qg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-plain-obj": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-3.0.0.tgz", + "integrity": "sha512-gwsOE28k+23GP1B6vFl1oVh/WOzmawBrKwo5Ev6wMKzPkaXaCDIQKzLnvsA42DRlbVTWorkgTKIviAKCWkfUwA==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-potential-custom-element-name": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", + "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==" + }, + "node_modules/is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-regexp": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-regexp/-/is-regexp-1.0.0.tgz", + "integrity": "sha512-7zjFAPO4/gwyQAAgRRmqeEeyIICSdmCqa3tsVHMdBzaXXRiqopZL4Cyghg/XulGWrtABTpbnYYzzIRffLkP4oA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-root": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-root/-/is-root-2.1.0.tgz", + "integrity": "sha512-AGOriNp96vNBd3HtU+RzFEc75FfR5ymiYv8E553I71SCeXBiMsVDUtdio1OEFvrPyLIQ9tVR5RxXIFe5PUFjMg==", + "engines": { + "node": ">=6" + } + }, + "node_modules/is-set": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.2.tgz", + "integrity": "sha512-+2cnTEZeY5z/iXGbLhPrOAaK/Mau5k5eXq9j14CpRTftq0pAJu2MwVRSZhyZWBzx3o6X795Lz6Bpb6R0GKf37g==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-shared-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", + "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "dependencies": { + "call-bind": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dependencies": { + "has-symbols": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-typed-array": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.12.tgz", + "integrity": "sha512-Z14TF2JNG8Lss5/HMqt0//T9JeHXttXy5pH/DBU4vi98ozO2btxzq9MwYDZYnKwU8nRsz/+GVFVRDq3DkVuSPg==", + "dependencies": { + "which-typed-array": "^1.1.11" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-typedarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", + "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==" + }, + "node_modules/is-weakmap": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.1.tgz", + "integrity": "sha512-NSBR4kH5oVj1Uwvv970ruUkCV7O1mzgVFO4/rev2cLRda9Tm9HrL70ZPut4rOHgY0FNrUu9BCbXA2sdQ+x0chA==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-weakref": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", + "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "dependencies": { + "call-bind": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-weakset": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.2.tgz", + "integrity": "sha512-t2yVvttHkQktwnNNmBQ98AhENLdPUTDTE21uPqAQ0ARwQfGeQKRVS0NNurH7bTf7RrvcVn1OOge45CnBeHCSmg==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-wsl": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "dependencies": { + "is-docker": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==" + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==" + }, + "node_modules/istanbul-lib-coverage": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.0.tgz", + "integrity": "sha512-eOeJ5BHCmHYvQK7xt9GkdHuzuCGS1Y6g9Gvnx3Ym33fz/HpLRYxiS0wHNr+m/MBC8B647Xt608vCDEvhl9c6Mw==", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.1.tgz", + "integrity": "sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==", + "dependencies": { + "@babel/core": "^7.12.3", + "@babel/parser": "^7.14.7", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", + "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-report/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-report/node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/istanbul-lib-report/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", + "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "dependencies": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/istanbul-reports": { + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.6.tgz", + "integrity": "sha512-TLgnMkKg3iTDsQ9PbPTdpfAK2DzjF9mqUG7RMgcQl8oFjad8ob4laGxv5XV5U9MAfx8D6tSJiUyuAwzLicaxlg==", + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/iterator.prototype": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/iterator.prototype/-/iterator.prototype-1.1.0.tgz", + "integrity": "sha512-rjuhAk1AJ1fssphHD0IFV6TWL40CwRZ53FrztKx43yk2v6rguBYsY4Bj1VU4HmoMmKwZUlx7mfnhDf9cOp4YTw==", + "dependencies": { + "define-properties": "^1.1.4", + "get-intrinsic": "^1.1.3", + "has-symbols": "^1.0.3", + "has-tostringtag": "^1.0.0", + "reflect.getprototypeof": "^1.0.3" + } + }, + "node_modules/jake": { + "version": "10.8.7", + "resolved": "https://registry.npmjs.org/jake/-/jake-10.8.7.tgz", + "integrity": "sha512-ZDi3aP+fG/LchyBzUM804VjddnwfSfsdeYkwt8NcbKRvo4rFkjhs456iLFn3k2ZUWvNe4i48WACDbza8fhq2+w==", + "dependencies": { + "async": "^3.2.3", + "chalk": "^4.0.2", + "filelist": "^1.0.4", + "minimatch": "^3.1.2" + }, + "bin": { + "jake": "bin/cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jake/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jake/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jake/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jake/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==" + }, + "node_modules/jake/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jake/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest/-/jest-27.5.1.tgz", + "integrity": "sha512-Yn0mADZB89zTtjkPJEXwrac3LHudkQMR+Paqa8uxJHCBr9agxztUifWCyiYrjhMPBoUVBjyny0I7XH6ozDr7QQ==", + "dependencies": { + "@jest/core": "^27.5.1", + "import-local": "^3.0.2", + "jest-cli": "^27.5.1" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-changed-files": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-27.5.1.tgz", + "integrity": "sha512-buBLMiByfWGCoMsLLzGUUSpAmIAGnbR2KJoMN10ziLhOLvP4e0SlypHnAel8iqQXTrcbmfEY9sSqae5sgUsTvw==", + "dependencies": { + "@jest/types": "^27.5.1", + "execa": "^5.0.0", + "throat": "^6.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-circus": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-27.5.1.tgz", + "integrity": "sha512-D95R7x5UtlMA5iBYsOHFFbMD/GVA4R/Kdq15f7xYWUfWHBto9NYRsOvnSauTgdF+ogCpJ4tyKOXhUifxS65gdw==", + "dependencies": { + "@jest/environment": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "dedent": "^0.7.0", + "expect": "^27.5.1", + "is-generator-fn": "^2.0.0", + "jest-each": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "pretty-format": "^27.5.1", + "slash": "^3.0.0", + "stack-utils": "^2.0.3", + "throat": "^6.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-circus/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-circus/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-circus/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-circus/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==" + }, + "node_modules/jest-circus/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-circus/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-cli": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-27.5.1.tgz", + "integrity": "sha512-Hc6HOOwYq4/74/c62dEE3r5elx8wjYqxY0r0G/nFrLDPMFRu6RA/u8qINOIkvhxG7mMQ5EJsOGfRpI8L6eFUVw==", + "dependencies": { + "@jest/core": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "import-local": "^3.0.2", + "jest-config": "^27.5.1", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "prompts": "^2.0.1", + "yargs": "^16.2.0" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-cli/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-cli/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-cli/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-cli/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==" + }, + "node_modules/jest-cli/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-cli/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-config": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-27.5.1.tgz", + "integrity": "sha512-5sAsjm6tGdsVbW9ahcChPAFCk4IlkQUknH5AvKjuLTSlcO/wCZKyFdn7Rg0EkC+OGgWODEy2hDpWB1PgzH0JNA==", + "dependencies": { + "@babel/core": "^7.8.0", + "@jest/test-sequencer": "^27.5.1", + "@jest/types": "^27.5.1", + "babel-jest": "^27.5.1", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.1", + "graceful-fs": "^4.2.9", + "jest-circus": "^27.5.1", + "jest-environment-jsdom": "^27.5.1", + "jest-environment-node": "^27.5.1", + "jest-get-type": "^27.5.1", + "jest-jasmine2": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-runner": "^27.5.1", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "micromatch": "^4.0.4", + "parse-json": "^5.2.0", + "pretty-format": "^27.5.1", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "peerDependencies": { + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "ts-node": { + "optional": true + } + } + }, + "node_modules/jest-config/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-config/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-config/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-config/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==" + }, + "node_modules/jest-config/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-config/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-diff": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-27.5.1.tgz", + "integrity": "sha512-m0NvkX55LDt9T4mctTEgnZk3fmEg3NRYutvMPWM/0iPnkFj2wIeF45O1718cMSOFO1vINkqmxqD8vE37uTEbqw==", + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^27.5.1", + "jest-get-type": "^27.5.1", + "pretty-format": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-diff/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-diff/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-diff/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-diff/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==" + }, + "node_modules/jest-diff/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-diff/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-docblock": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-27.5.1.tgz", + "integrity": "sha512-rl7hlABeTsRYxKiUfpHrQrG4e2obOiTQWfMEH3PxPjOtdsfLQO4ReWSZaQ7DETm4xu07rl4q/h4zcKXyU0/OzQ==", + "dependencies": { + "detect-newline": "^3.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-each": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-27.5.1.tgz", + "integrity": "sha512-1Ff6p+FbhT/bXQnEouYy00bkNSY7OUpfIcmdl8vZ31A1UUaurOLPA8a8BbJOF2RDUElwJhmeaV7LnagI+5UwNQ==", + "dependencies": { + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "jest-get-type": "^27.5.1", + "jest-util": "^27.5.1", + "pretty-format": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-each/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-each/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-each/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-each/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==" + }, + "node_modules/jest-each/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-each/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-environment-jsdom": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-environment-jsdom/-/jest-environment-jsdom-27.5.1.tgz", + "integrity": "sha512-TFBvkTC1Hnnnrka/fUb56atfDtJ9VMZ94JkjTbggl1PEpwrYtUBKMezB3inLmWqQsXYLcMwNoDQwoBTAvFfsfw==", + "dependencies": { + "@jest/environment": "^27.5.1", + "@jest/fake-timers": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "jest-mock": "^27.5.1", + "jest-util": "^27.5.1", + "jsdom": "^16.6.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-environment-node": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-27.5.1.tgz", + "integrity": "sha512-Jt4ZUnxdOsTGwSRAfKEnE6BcwsSPNOijjwifq5sDFSA2kesnXTvNqKHYgM0hDq3549Uf/KzdXNYn4wMZJPlFLw==", + "dependencies": { + "@jest/environment": "^27.5.1", + "@jest/fake-timers": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "jest-mock": "^27.5.1", + "jest-util": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-get-type": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-27.5.1.tgz", + "integrity": "sha512-2KY95ksYSaK7DMBWQn6dQz3kqAf3BB64y2udeG+hv4KfSOb9qwcYQstTJc1KCbsix+wLZWZYN8t7nwX3GOBLRw==", + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-haste-map": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-27.5.1.tgz", + "integrity": "sha512-7GgkZ4Fw4NFbMSDSpZwXeBiIbx+t/46nJ2QitkOjvwPYyZmqttu2TDSimMHP1EkPOi4xUZAN1doE5Vd25H4Jng==", + "dependencies": { + "@jest/types": "^27.5.1", + "@types/graceful-fs": "^4.1.2", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^27.5.1", + "jest-serializer": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", + "micromatch": "^4.0.4", + "walker": "^1.0.7" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.2" + } + }, + "node_modules/jest-jasmine2": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-jasmine2/-/jest-jasmine2-27.5.1.tgz", + "integrity": "sha512-jtq7VVyG8SqAorDpApwiJJImd0V2wv1xzdheGHRGyuT7gZm6gG47QEskOlzsN1PG/6WNaCo5pmwMHDf3AkG2pQ==", + "dependencies": { + "@jest/environment": "^27.5.1", + "@jest/source-map": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "expect": "^27.5.1", + "is-generator-fn": "^2.0.0", + "jest-each": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "pretty-format": "^27.5.1", + "throat": "^6.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-jasmine2/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-jasmine2/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-jasmine2/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-jasmine2/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==" + }, + "node_modules/jest-jasmine2/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-jasmine2/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-leak-detector": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-27.5.1.tgz", + "integrity": "sha512-POXfWAMvfU6WMUXftV4HolnJfnPOGEu10fscNCA76KBpRRhcMN2c8d3iT2pxQS3HLbA+5X4sOUPzYO2NUyIlHQ==", + "dependencies": { + "jest-get-type": "^27.5.1", + "pretty-format": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-matcher-utils": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-27.5.1.tgz", + "integrity": "sha512-z2uTx/T6LBaCoNWNFWwChLBKYxTMcGBRjAt+2SbP929/Fflb9aa5LGma654Rz8z9HLxsrUaYzxE9T/EFIL/PAw==", + "dependencies": { + "chalk": "^4.0.0", + "jest-diff": "^27.5.1", + "jest-get-type": "^27.5.1", + "pretty-format": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-matcher-utils/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-matcher-utils/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-matcher-utils/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-matcher-utils/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==" + }, + "node_modules/jest-matcher-utils/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-matcher-utils/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-message-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-27.5.1.tgz", + "integrity": "sha512-rMyFe1+jnyAAf+NHwTclDz0eAaLkVDdKVHHBFWsBWHnnh5YeJMNWWsv7AbFYXfK3oTqvL7VTWkhNLu1jX24D+g==", + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^27.5.1", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^27.5.1", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-message-util/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-message-util/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-message-util/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-message-util/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==" + }, + "node_modules/jest-message-util/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-message-util/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-mock": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-27.5.1.tgz", + "integrity": "sha512-K4jKbY1d4ENhbrG2zuPWaQBvDly+iZ2yAW+T1fATN78hc0sInwn7wZB8XtlNnvHug5RMwV897Xm4LqmPM4e2Og==", + "dependencies": { + "@jest/types": "^27.5.1", + "@types/node": "*" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-pnp-resolver": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.3.tgz", + "integrity": "sha512-+3NpwQEnRoIBtx4fyhblQDPgJI0H1IEIkX7ShLUjPGA7TtUTvI1oiKi3SR4oBR0hQhQR80l4WAe5RrXBwWMA8w==", + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "jest-resolve": "*" + }, + "peerDependenciesMeta": { + "jest-resolve": { + "optional": true + } + } + }, + "node_modules/jest-regex-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-27.5.1.tgz", + "integrity": "sha512-4bfKq2zie+x16okqDXjXn9ql2B0dScQu+vcwe4TvFVhkVyuWLqpZrZtXxLLWoXYgn0E87I6r6GRYHF7wFZBUvg==", + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-resolve": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-27.5.1.tgz", + "integrity": "sha512-FFDy8/9E6CV83IMbDpcjOhumAQPDyETnU2KZ1O98DwTnz8AOBsW/Xv3GySr1mOZdItLR+zDZ7I/UdTFbgSOVCw==", + "dependencies": { + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "resolve": "^1.20.0", + "resolve.exports": "^1.1.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-resolve-dependencies": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-27.5.1.tgz", + "integrity": "sha512-QQOOdY4PE39iawDn5rzbIePNigfe5B9Z91GDD1ae/xNDlu9kaat8QQ5EKnNmVWPV54hUdxCVwwj6YMgR2O7IOg==", + "dependencies": { + "@jest/types": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-snapshot": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-resolve/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-resolve/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-resolve/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-resolve/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==" + }, + "node_modules/jest-resolve/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-resolve/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-runner": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-27.5.1.tgz", + "integrity": "sha512-g4NPsM4mFCOwFKXO4p/H/kWGdJp9V8kURY2lX8Me2drgXqG7rrZAx5kv+5H7wtt/cdFIjhqYx1HrlqWHaOvDaQ==", + "dependencies": { + "@jest/console": "^27.5.1", + "@jest/environment": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.8.1", + "graceful-fs": "^4.2.9", + "jest-docblock": "^27.5.1", + "jest-environment-jsdom": "^27.5.1", + "jest-environment-node": "^27.5.1", + "jest-haste-map": "^27.5.1", + "jest-leak-detector": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", + "source-map-support": "^0.5.6", + "throat": "^6.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-runner/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-runner/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-runner/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-runner/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==" + }, + "node_modules/jest-runner/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-runner/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-runtime": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-27.5.1.tgz", + "integrity": "sha512-o7gxw3Gf+H2IGt8fv0RiyE1+r83FJBRruoA+FXrlHw6xEyBsU8ugA6IPfTdVyA0w8HClpbK+DGJxH59UrNMx8A==", + "dependencies": { + "@jest/environment": "^27.5.1", + "@jest/fake-timers": "^27.5.1", + "@jest/globals": "^27.5.1", + "@jest/source-map": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "cjs-module-lexer": "^1.0.0", + "collect-v8-coverage": "^1.0.0", + "execa": "^5.0.0", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-mock": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "slash": "^3.0.0", + "strip-bom": "^4.0.0" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-runtime/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-runtime/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-runtime/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-runtime/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==" + }, + "node_modules/jest-runtime/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-runtime/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-serializer": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-27.5.1.tgz", + "integrity": "sha512-jZCyo6iIxO1aqUxpuBlwTDMkzOAJS4a3eYz3YzgxxVQFwLeSA7Jfq5cbqCY+JLvTDrWirgusI/0KwxKMgrdf7w==", + "dependencies": { + "@types/node": "*", + "graceful-fs": "^4.2.9" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-snapshot": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-27.5.1.tgz", + "integrity": "sha512-yYykXI5a0I31xX67mgeLw1DZ0bJB+gpq5IpSuCAoyDi0+BhgU/RIrL+RTzDmkNTchvDFWKP8lp+w/42Z3us5sA==", + "dependencies": { + "@babel/core": "^7.7.2", + "@babel/generator": "^7.7.2", + "@babel/plugin-syntax-typescript": "^7.7.2", + "@babel/traverse": "^7.7.2", + "@babel/types": "^7.0.0", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/babel__traverse": "^7.0.4", + "@types/prettier": "^2.1.5", + "babel-preset-current-node-syntax": "^1.0.0", + "chalk": "^4.0.0", + "expect": "^27.5.1", + "graceful-fs": "^4.2.9", + "jest-diff": "^27.5.1", + "jest-get-type": "^27.5.1", + "jest-haste-map": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-util": "^27.5.1", + "natural-compare": "^1.4.0", + "pretty-format": "^27.5.1", + "semver": "^7.3.2" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-snapshot/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-snapshot/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-snapshot/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-snapshot/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==" + }, + "node_modules/jest-snapshot/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-snapshot/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-27.5.1.tgz", + "integrity": "sha512-Kv2o/8jNvX1MQ0KGtw480E/w4fBCDOnH6+6DmeKi6LZUIlKA5kwY0YNdlzaWTiVgxqAqik11QyxDOKk543aKXw==", + "dependencies": { + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-util/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-util/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-util/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-util/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==" + }, + "node_modules/jest-util/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-util/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-validate": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-27.5.1.tgz", + "integrity": "sha512-thkNli0LYTmOI1tDB3FI1S1RTp/Bqyd9pTarJwL87OIBFuqEb5Apv5EaApEudYg4g86e3CT6kM0RowkhtEnCBQ==", + "dependencies": { + "@jest/types": "^27.5.1", + "camelcase": "^6.2.0", + "chalk": "^4.0.0", + "jest-get-type": "^27.5.1", + "leven": "^3.1.0", + "pretty-format": "^27.5.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-validate/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-validate/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-validate/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-validate/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==" + }, + "node_modules/jest-validate/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-validate/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-watch-typeahead": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/jest-watch-typeahead/-/jest-watch-typeahead-1.1.0.tgz", + "integrity": "sha512-Va5nLSJTN7YFtC2jd+7wsoe1pNe5K4ShLux/E5iHEwlB9AxaxmggY7to9KUqKojhaJw3aXqt5WAb4jGPOolpEw==", + "dependencies": { + "ansi-escapes": "^4.3.1", + "chalk": "^4.0.0", + "jest-regex-util": "^28.0.0", + "jest-watcher": "^28.0.0", + "slash": "^4.0.0", + "string-length": "^5.0.1", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "peerDependencies": { + "jest": "^27.0.0 || ^28.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/@jest/console": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-28.1.3.tgz", + "integrity": "sha512-QPAkP5EwKdK/bxIr6C1I4Vs0rm2nHiANzj/Z5X2JQkrZo6IqvC4ldZ9K95tF0HdidhA8Bo6egxSzUFPYKcEXLw==", + "dependencies": { + "@jest/types": "^28.1.3", + "@types/node": "*", + "chalk": "^4.0.0", + "jest-message-util": "^28.1.3", + "jest-util": "^28.1.3", + "slash": "^3.0.0" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/@jest/console/node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-watch-typeahead/node_modules/@jest/test-result": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-28.1.3.tgz", + "integrity": "sha512-kZAkxnSE+FqE8YjW8gNuoVkkC9I7S1qmenl8sGcDOLropASP+BkcGKwhXoyqQuGOGeYY0y/ixjrd/iERpEXHNg==", + "dependencies": { + "@jest/console": "^28.1.3", + "@jest/types": "^28.1.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "collect-v8-coverage": "^1.0.0" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/@jest/types": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-28.1.3.tgz", + "integrity": "sha512-RyjiyMUZrKz/c+zlMFO1pm70DcIlST8AeWTkoUdZevew44wcNZQHsEVOiCVtgVnlFFD82FPaXycys58cf2muVQ==", + "dependencies": { + "@jest/schemas": "^28.1.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^17.0.8", + "chalk": "^4.0.0" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/@types/yargs": { + "version": "17.0.24", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.24.tgz", + "integrity": "sha512-6i0aC7jV6QzQB8ne1joVZ0eSFIstHsCrobmOtghM11yGlH0j43FKL2UhWdELkyps0zuf7qVTUVCCR+tgSlyLLw==", + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/jest-watch-typeahead/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-watch-typeahead/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-watch-typeahead/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-watch-typeahead/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==" + }, + "node_modules/jest-watch-typeahead/node_modules/emittery": { + "version": "0.10.2", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.10.2.tgz", + "integrity": "sha512-aITqOwnLanpHLNXZJENbOgjUBeHocD+xsSJmNrjovKBW5HbSpW3d1pEls7GFQPUWXiwG9+0P4GtHfEqC/4M0Iw==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "node_modules/jest-watch-typeahead/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-watch-typeahead/node_modules/jest-message-util": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-28.1.3.tgz", + "integrity": "sha512-PFdn9Iewbt575zKPf1286Ht9EPoJmYT7P0kY+RibeYZ2XtOr53pDLEFoTWXbd1h4JiGiWpTBC84fc8xMXQMb7g==", + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^28.1.3", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^28.1.3", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/jest-message-util/node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-watch-typeahead/node_modules/jest-regex-util": { + "version": "28.0.2", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-28.0.2.tgz", + "integrity": "sha512-4s0IgyNIy0y9FK+cjoVYoxamT7Zeo7MhzqRGx7YDYmaQn1wucY9rotiGkBzzcMXTtjrCAP/f7f+E0F7+fxPNdw==", + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/jest-util": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-28.1.3.tgz", + "integrity": "sha512-XdqfpHwpcSRko/C35uLYFM2emRAltIIKZiJ9eAmhjsj0CqZMa0p1ib0R5fWIqGhn1a103DebTbpqIaP1qCQ6tQ==", + "dependencies": { + "@jest/types": "^28.1.3", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/jest-watcher": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-28.1.3.tgz", + "integrity": "sha512-t4qcqj9hze+jviFPUN3YAtAEeFnr/azITXQEMARf5cMwKY2SMBRnCQTXLixTl20OR6mLh9KLMrgVJgJISym+1g==", + "dependencies": { + "@jest/test-result": "^28.1.3", + "@jest/types": "^28.1.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "emittery": "^0.10.2", + "jest-util": "^28.1.3", + "string-length": "^4.0.1" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/jest-watcher/node_modules/string-length": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", + "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jest-watch-typeahead/node_modules/jest-watcher/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-watch-typeahead/node_modules/pretty-format": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-28.1.3.tgz", + "integrity": "sha512-8gFb/To0OmxHR9+ZTb14Df2vNxdGCX8g1xWGUTqUw5TiZvcQf5sHKObd5UcPyLLyowNwDAMTF3XWOG1B6mxl1Q==", + "dependencies": { + "@jest/schemas": "^28.1.3", + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || ^16.10.0 || >=17.0.0" + } + }, + "node_modules/jest-watch-typeahead/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-watch-typeahead/node_modules/react-is": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.2.0.tgz", + "integrity": "sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==" + }, + "node_modules/jest-watch-typeahead/node_modules/slash": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-4.0.0.tgz", + "integrity": "sha512-3dOsAHXXUkQTpOYcoAxLIorMTp4gIQr5IW3iVb7A7lFIp0VHhnynm9izx6TssdrIcVIESAlVjtnO2K8bg+Coew==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/jest-watch-typeahead/node_modules/string-length": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-5.0.1.tgz", + "integrity": "sha512-9Ep08KAMUn0OadnVaBuRdE2l615CQ508kr0XMadjClfYpdCyvrbFp6Taebo8yyxokQ4viUd/xPPUA4FGgUa0ow==", + "dependencies": { + "char-regex": "^2.0.0", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12.20" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/jest-watch-typeahead/node_modules/string-length/node_modules/char-regex": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-2.0.1.tgz", + "integrity": "sha512-oSvEeo6ZUD7NepqAat3RqoucZ5SeqLJgOvVIwkafu6IP3V0pO38s/ypdVUmDDK6qIIHNlYHJAKX9E7R7HoKElw==", + "engines": { + "node": ">=12.20" + } + }, + "node_modules/jest-watch-typeahead/node_modules/strip-ansi": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", + "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", + "dependencies": { + "ansi-regex": "^6.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/strip-ansi?sponsor=1" + } + }, + "node_modules/jest-watch-typeahead/node_modules/strip-ansi/node_modules/ansi-regex": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz", + "integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-regex?sponsor=1" + } + }, + "node_modules/jest-watch-typeahead/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-watcher": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-27.5.1.tgz", + "integrity": "sha512-z676SuD6Z8o8qbmEGhoEUFOM1+jfEiL3DXHK/xgEiG2EyNYfFG60jluWcupY6dATjfEsKQuibReS1djInQnoVw==", + "dependencies": { + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "jest-util": "^27.5.1", + "string-length": "^4.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/jest-watcher/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-watcher/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/jest-watcher/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/jest-watcher/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==" + }, + "node_modules/jest-watcher/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-watcher/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-worker": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", + "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", + "dependencies": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/jest-worker/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-worker/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/jiti": { + "version": "1.19.3", + "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.19.3.tgz", + "integrity": "sha512-5eEbBDQT/jF1xg6l36P+mWGGoH9Spuy0PCdSr2dtWRDGC6ph/w9ZCL4lmESW8f8F7MwT3XKescfP0wnZWAKL9w==", + "bin": { + "jiti": "bin/jiti.js" + } + }, + "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": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsdom": { + "version": "16.7.0", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-16.7.0.tgz", + "integrity": "sha512-u9Smc2G1USStM+s/x1ru5Sxrl6mPYCbByG1U/hUmqaVsm4tbNyS7CicOSRyuGQYZhTu0h84qkZZQ/I+dzizSVw==", + "dependencies": { + "abab": "^2.0.5", + "acorn": "^8.2.4", + "acorn-globals": "^6.0.0", + "cssom": "^0.4.4", + "cssstyle": "^2.3.0", + "data-urls": "^2.0.0", + "decimal.js": "^10.2.1", + "domexception": "^2.0.1", + "escodegen": "^2.0.0", + "form-data": "^3.0.0", + "html-encoding-sniffer": "^2.0.1", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-potential-custom-element-name": "^1.0.1", + "nwsapi": "^2.2.0", + "parse5": "6.0.1", + "saxes": "^5.0.1", + "symbol-tree": "^3.2.4", + "tough-cookie": "^4.0.0", + "w3c-hr-time": "^1.0.2", + "w3c-xmlserializer": "^2.0.0", + "webidl-conversions": "^6.1.0", + "whatwg-encoding": "^1.0.5", + "whatwg-mimetype": "^2.3.0", + "whatwg-url": "^8.5.0", + "ws": "^7.4.6", + "xml-name-validator": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "canvas": "^2.5.0" + }, + "peerDependenciesMeta": { + "canvas": { + "optional": true + } + } + }, + "node_modules/jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=4" + } + }, + "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==" + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==" + }, + "node_modules/json-schema": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz", + "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==" + }, + "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==" + }, + "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==" + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/jsonpointer": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/jsonpointer/-/jsonpointer-5.0.1.tgz", + "integrity": "sha512-p/nXbhSEcu3pZRdkW1OfJhpsVtW1gd4Wa1fnQc9YLiTfAjn0312eMKimbdIQzuZl9aa9xUGaRlP9T/CJE/ditQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/jsx-ast-utils": { + "version": "3.3.5", + "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.5.tgz", + "integrity": "sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==", + "dependencies": { + "array-includes": "^3.1.6", + "array.prototype.flat": "^1.3.1", + "object.assign": "^4.1.4", + "object.values": "^1.1.6" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/jwt-decode": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-4.0.0.tgz", + "integrity": "sha512-+KJGIyHgkGuIq3IEBNftfhW/LfWhXUIY6OmyVWjliu5KH1y0fw7VQ8YndE2O4qZdMSd9SqbnC8GOcZEy0Om7sA==", + "engines": { + "node": ">=18" + } + }, + "node_modules/keyv": { + "version": "4.5.3", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.3.tgz", + "integrity": "sha512-QCiSav9WaX1PgETJ+SpNnx2PRRapJ/oRSXM4VO5OGYGSjrxbKPVFVhB3l2OCbLCk329N8qyAtsJjSjvVBWzEug==", + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "engines": { + "node": ">=6" + } + }, + "node_modules/klona": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/klona/-/klona-2.0.6.tgz", + "integrity": "sha512-dhG34DXATL5hSxJbIexCft8FChFXtmskoZYnoPWjXQuebWYCNkVeV3KkGegCK9CP1oswI/vQibS2GY7Em/sJJA==", + "engines": { + "node": ">= 8" + } + }, + "node_modules/language-subtag-registry": { + "version": "0.3.22", + "resolved": "https://registry.npmjs.org/language-subtag-registry/-/language-subtag-registry-0.3.22.tgz", + "integrity": "sha512-tN0MCzyWnoz/4nHS6uxdlFWoUZT7ABptwKPQ52Ea7URk6vll88bWBVhodtnlfEuCcKWNGoc+uGbw1cwa9IKh/w==" + }, + "node_modules/language-tags": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/language-tags/-/language-tags-1.0.5.tgz", + "integrity": "sha512-qJhlO9cGXi6hBGKoxEG/sKZDAHD5Hnu9Hs4WbOY3pCWXDhw0N8x1NenNzm2EnNLkLkk7J2SdxAkDSbb6ftT+UQ==", + "dependencies": { + "language-subtag-registry": "~0.3.2" + } + }, + "node_modules/launch-editor": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/launch-editor/-/launch-editor-2.6.0.tgz", + "integrity": "sha512-JpDCcQnyAAzZZaZ7vEiSqL690w7dAEyLao+KC96zBplnYbJS7TYNjvM3M7y3dGz+v7aIsJk3hllWuc0kWAjyRQ==", + "dependencies": { + "picocolors": "^1.0.0", + "shell-quote": "^1.7.3" + } + }, + "node_modules/leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "engines": { + "node": ">=6" + } + }, + "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==", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/lilconfig": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.1.0.tgz", + "integrity": "sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ==", + "engines": { + "node": ">=10" + } + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==" + }, + "node_modules/loader-runner": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-4.3.0.tgz", + "integrity": "sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg==", + "engines": { + "node": ">=6.11.5" + } + }, + "node_modules/loader-utils": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-2.0.4.tgz", + "integrity": "sha512-xXqpXoINfFhgua9xiqD8fPFHgkoq1mmmpE92WlDbm9rNRd/EbRb+Gqf908T2DMfuHjjJlksiK2RbHVOdD/MqSw==", + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + }, + "engines": { + "node": ">=8.9.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==", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, + "node_modules/lodash.debounce": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz", + "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==" + }, + "node_modules/lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==" + }, + "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==" + }, + "node_modules/lodash.sortby": { + "version": "4.7.0", + "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", + "integrity": "sha512-HDWXG8isMntAyRF5vZ7xKuEvOhT4AhlRt/3czTSjvGUxjYCBVRQY48ViDHyfYz9VIoBkW4TMGQNapx+l3RUwdA==" + }, + "node_modules/lodash.uniq": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", + "integrity": "sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==" + }, + "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/lower-case": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz", + "integrity": "sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==", + "dependencies": { + "tslib": "^2.0.3" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/lz-string": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.5.0.tgz", + "integrity": "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==", + "bin": { + "lz-string": "bin/bin.js" + } + }, + "node_modules/magic-string": { + "version": "0.25.9", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.9.tgz", + "integrity": "sha512-RmF0AsMzgt25qzqqLc1+MbHmhdx0ojF2Fvs4XnOqz2ZOBXzzkEwc/dJQZCYHAn7v1jbVOjAZfK8msRn4BxO4VQ==", + "dependencies": { + "sourcemap-codec": "^1.4.8" + } + }, + "node_modules/make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dependencies": { + "semver": "^6.0.0" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-dir/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dependencies": { + "tmpl": "1.0.5" + } + }, + "node_modules/mdn-data": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.4.tgz", + "integrity": "sha512-iV3XNKw06j5Q7mi6h+9vbx23Tv7JkjEVgKHW4pimwyDGWm0OIQntJJ+u1C6mg6mK1EaTv42XQ7w76yuzH7M2cA==" + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/memfs": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/memfs/-/memfs-3.5.3.tgz", + "integrity": "sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==", + "dependencies": { + "fs-monkey": "^1.0.4" + }, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==" + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==" + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "engines": { + "node": ">= 8" + } + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dependencies": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "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/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "engines": { + "node": ">=6" + } + }, + "node_modules/min-indent": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/min-indent/-/min-indent-1.0.1.tgz", + "integrity": "sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==", + "engines": { + "node": ">=4" + } + }, + "node_modules/mini-css-extract-plugin": { + "version": "2.7.6", + "resolved": "https://registry.npmjs.org/mini-css-extract-plugin/-/mini-css-extract-plugin-2.7.6.tgz", + "integrity": "sha512-Qk7HcgaPkGG6eD77mLvZS1nmxlao3j+9PkrT9Uc7HAE1id3F41+DdBRYRYkbyfNRGzm8/YWtzhw7nVPmwhqTQw==", + "dependencies": { + "schema-utils": "^4.0.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.0.0" + } + }, + "node_modules/mini-css-extract-plugin/node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/mini-css-extract-plugin/node_modules/ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dependencies": { + "fast-deep-equal": "^3.1.3" + }, + "peerDependencies": { + "ajv": "^8.8.2" + } + }, + "node_modules/mini-css-extract-plugin/node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/mini-css-extract-plugin/node_modules/schema-utils": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.2.0.tgz", + "integrity": "sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/minimalistic-assert": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", + "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==" + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + }, + "node_modules/multicast-dns": { + "version": "7.2.5", + "resolved": "https://registry.npmjs.org/multicast-dns/-/multicast-dns-7.2.5.tgz", + "integrity": "sha512-2eznPJP8z2BFLX50tf0LuODrpINqP1RVIm/CObbTcBRITQgmC/TjcREF1NeTBzIcR5XO/ukWo+YHOjBbFwIupg==", + "dependencies": { + "dns-packet": "^5.2.2", + "thunky": "^1.0.2" + }, + "bin": { + "multicast-dns": "cli.js" + } + }, + "node_modules/mz": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz", + "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==", + "dependencies": { + "any-promise": "^1.0.0", + "object-assign": "^4.0.1", + "thenify-all": "^1.0.0" + } + }, + "node_modules/nanoid": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-5.0.1.tgz", + "integrity": "sha512-vWeVtV5Cw68aML/QaZvqN/3QQXc6fBfIieAlu05m7FZW2Dgb+3f0xc0TTxuJW+7u30t7iSDTV/j3kVI0oJqIfQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "bin": { + "nanoid": "bin/nanoid.js" + }, + "engines": { + "node": "^18 || >=20" + } + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==" + }, + "node_modules/natural-compare-lite": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare-lite/-/natural-compare-lite-1.4.0.tgz", + "integrity": "sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==" + }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==" + }, + "node_modules/no-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.4.tgz", + "integrity": "sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==", + "dependencies": { + "lower-case": "^2.0.2", + "tslib": "^2.0.3" + } + }, + "node_modules/node-forge": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-1.3.1.tgz", + "integrity": "sha512-dPEtOeMvF9VMcYV/1Wb8CPoVAXtp6MKMlcbAt4ddqmGqUJ6fQZFXkNZNkNlfevtNkGtaSoXf/vNNNSvgrdXwtA==", + "engines": { + "node": ">= 6.13.0" + } + }, + "node_modules/node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==" + }, + "node_modules/node-releases": { + "version": "2.0.13", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.13.tgz", + "integrity": "sha512-uYr7J37ae/ORWdZeQ1xxMJe3NtdmqMC/JZK+geofDrkLUApKRHPd18/TxtBOJ4A0/+uUIliorNrfYV6s1b02eQ==" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/normalize-range": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", + "integrity": "sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/normalize-url": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-6.1.0.tgz", + "integrity": "sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/nth-check": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz", + "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==", + "dependencies": { + "boolbase": "^1.0.0" + }, + "funding": { + "url": "https://github.com/fb55/nth-check?sponsor=1" + } + }, + "node_modules/nwsapi": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.7.tgz", + "integrity": "sha512-ub5E4+FBPKwAZx0UwIQOjYWGHTEq5sPqHQNRN8Z9e4A7u3Tj1weLJsL59yH9vmvqEtBHaOmT6cYQKIZOxp35FQ==" + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-hash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-3.0.0.tgz", + "integrity": "sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/object-inspect": { + "version": "1.12.3", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz", + "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object-is": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/object-is/-/object-is-1.1.5.tgz", + "integrity": "sha512-3cyDsyHgtmi7I7DfSSI2LDp6SK2lwvtbg0p0R1e0RvTqF5ceGx+K2dfSjm1bKDMVCFEDAQvy+o8c6a7VujOddw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.assign": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", + "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "has-symbols": "^1.0.3", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.entries": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.6.tgz", + "integrity": "sha512-leTPzo4Zvg3pmbQ3rDK69Rl8GQvIqMWubrkxONG9/ojtFE2rD9fjMKfSI5BxW3osRH1m6VdzmqK8oAY9aT4x5w==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.fromentries": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.6.tgz", + "integrity": "sha512-VciD13dswC4j1Xt5394WR4MzmAQmlgN72phd/riNp9vtD7tp4QQWJ0R4wvclXcafgcYK8veHRed2W6XeGBvcfg==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.getownpropertydescriptors": { + "version": "2.1.6", + "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.6.tgz", + "integrity": "sha512-lq+61g26E/BgHv0ZTFgRvi7NMEPuAxLkFU7rukXjc/AlwH4Am5xXVnIXy3un1bg/JPbXHrixRkK1itUzzPiIjQ==", + "dependencies": { + "array.prototype.reduce": "^1.0.5", + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.21.2", + "safe-array-concat": "^1.0.0" + }, + "engines": { + "node": ">= 0.8" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.groupby": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.0.tgz", + "integrity": "sha512-70MWG6NfRH9GnbZOikuhPPYzpUpof9iW2J9E4dW7FXTqPNb6rllE6u39SKwwiNh8lCwX3DDb5OgcKGiEBrTTyw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.21.2", + "get-intrinsic": "^1.2.1" + } + }, + "node_modules/object.hasown": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/object.hasown/-/object.hasown-1.1.2.tgz", + "integrity": "sha512-B5UIT3J1W+WuWIU55h0mjlwaqxiE5vYENJXIXZ4VFe05pNYrkKuK0U/6aFcb0pKywYJh7IhfoqUfKVmrJJHZHw==", + "dependencies": { + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.values": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.6.tgz", + "integrity": "sha512-FVVTkD1vENCsAcwNs9k6jea2uHC/X0+JcjG8YA60FN5CMaJmG95wT9jek/xX9nornqGRrBkKtzuAu2wuHpKqvw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/obuf": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/obuf/-/obuf-1.1.2.tgz", + "integrity": "sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==" + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/on-headers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", + "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/open": { + "version": "8.4.2", + "resolved": "https://registry.npmjs.org/open/-/open-8.4.2.tgz", + "integrity": "sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==", + "dependencies": { + "define-lazy-prop": "^2.0.0", + "is-docker": "^2.1.1", + "is-wsl": "^2.2.0" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/optionator": { + "version": "0.9.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", + "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", + "dependencies": { + "@aashutoshrathi/word-wrap": "^1.2.3", + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0" + }, + "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==", + "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==", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-retry": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/p-retry/-/p-retry-4.6.2.tgz", + "integrity": "sha512-312Id396EbJdvRONlngUx0NydfrIQ5lsYu0znKVUzVvArzEIt08V1qhtyESbGVd1FGX7UKtiFp5uwKZdM8wIuQ==", + "dependencies": { + "@types/retry": "0.12.0", + "retry": "^0.13.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/param-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/param-case/-/param-case-3.0.4.tgz", + "integrity": "sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A==", + "dependencies": { + "dot-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "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==", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parse5": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz", + "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==" + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/pascal-case": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-3.1.2.tgz", + "integrity": "sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==", + "dependencies": { + "no-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "engines": { + "node": ">=0.10.0" + } + }, + "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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==" + }, + "node_modules/path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==" + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "engines": { + "node": ">=8" + } + }, + "node_modules/performance-now": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", + "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==" + }, + "node_modules/picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==" + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pirates": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.6.tgz", + "integrity": "sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/pkg-dir/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-up": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/pkg-up/-/pkg-up-3.1.0.tgz", + "integrity": "sha512-nDywThFk1i4BQK4twPQ6TA4RT8bDY96yeuCVBWL3ePARCiEKDRSrNGbFIgUJpLp+XeIR65v8ra7WuJOFUBtkMA==", + "dependencies": { + "find-up": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-up/node_modules/find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dependencies": { + "locate-path": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/pkg-up/node_modules/locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dependencies": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/pkg-up/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/pkg-up/node_modules/p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dependencies": { + "p-limit": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/pkg-up/node_modules/path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==", + "engines": { + "node": ">=4" + } + }, + "node_modules/postcss": { + "version": "8.4.28", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.28.tgz", + "integrity": "sha512-Z7V5j0cq8oEKyejIKfpD8b4eBy9cwW2JWPk0+fB1HOAMsfHbnAXLLS+PfVWlzMSLQaWttKDt607I0XHmpE67Vw==", + "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.6", + "picocolors": "^1.0.0", + "source-map-js": "^1.0.2" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/postcss-attribute-case-insensitive": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-attribute-case-insensitive/-/postcss-attribute-case-insensitive-5.0.2.tgz", + "integrity": "sha512-XIidXV8fDr0kKt28vqki84fRK8VW8eTuIa4PChv2MqKuT6C9UjmSKzen6KaWhWEoYvwxFCa7n/tC1SZ3tyq4SQ==", + "dependencies": { + "postcss-selector-parser": "^6.0.10" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-browser-comments": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-browser-comments/-/postcss-browser-comments-4.0.0.tgz", + "integrity": "sha512-X9X9/WN3KIvY9+hNERUqX9gncsgBA25XaeR+jshHz2j8+sYyHktHw1JdKuMjeLpGktXidqDhA7b/qm1mrBDmgg==", + "engines": { + "node": ">=8" + }, + "peerDependencies": { + "browserslist": ">=4", + "postcss": ">=8" + } + }, + "node_modules/postcss-calc": { + "version": "8.2.4", + "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-8.2.4.tgz", + "integrity": "sha512-SmWMSJmB8MRnnULldx0lQIyhSNvuDl9HfrZkaqqE/WHAhToYsAvDq+yAsA/kIyINDszOp3Rh0GFoNuH5Ypsm3Q==", + "dependencies": { + "postcss-selector-parser": "^6.0.9", + "postcss-value-parser": "^4.2.0" + }, + "peerDependencies": { + "postcss": "^8.2.2" + } + }, + "node_modules/postcss-clamp": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/postcss-clamp/-/postcss-clamp-4.1.0.tgz", + "integrity": "sha512-ry4b1Llo/9zz+PKC+030KUnPITTJAHeOwjfAyyB60eT0AorGLdzp52s31OsPRHRf8NchkgFoG2y6fCfn1IV1Ow==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": ">=7.6.0" + }, + "peerDependencies": { + "postcss": "^8.4.6" + } + }, + "node_modules/postcss-color-functional-notation": { + "version": "4.2.4", + "resolved": "https://registry.npmjs.org/postcss-color-functional-notation/-/postcss-color-functional-notation-4.2.4.tgz", + "integrity": "sha512-2yrTAUZUab9s6CpxkxC4rVgFEVaR6/2Pipvi6qcgvnYiVqZcbDHEoBDhrXzyb7Efh2CCfHQNtcqWcIruDTIUeg==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-color-hex-alpha": { + "version": "8.0.4", + "resolved": "https://registry.npmjs.org/postcss-color-hex-alpha/-/postcss-color-hex-alpha-8.0.4.tgz", + "integrity": "sha512-nLo2DCRC9eE4w2JmuKgVA3fGL3d01kGq752pVALF68qpGLmx2Qrk91QTKkdUqqp45T1K1XV8IhQpcu1hoAQflQ==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/postcss-color-rebeccapurple": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/postcss-color-rebeccapurple/-/postcss-color-rebeccapurple-7.1.1.tgz", + "integrity": "sha512-pGxkuVEInwLHgkNxUc4sdg4g3py7zUeCQ9sMfwyHAT+Ezk8a4OaaVZ8lIY5+oNqA/BXXgLyXv0+5wHP68R79hg==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-colormin": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-5.3.1.tgz", + "integrity": "sha512-UsWQG0AqTFQmpBegeLLc1+c3jIqBNB0zlDGRWR+dQ3pRKJL1oeMzyqmH3o2PIfn9MBdNrVPWhDbT769LxCTLJQ==", + "dependencies": { + "browserslist": "^4.21.4", + "caniuse-api": "^3.0.0", + "colord": "^2.9.1", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-convert-values": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-5.1.3.tgz", + "integrity": "sha512-82pC1xkJZtcJEfiLw6UXnXVXScgtBrjlO5CBmuDQc+dlb88ZYheFsjTn40+zBVi3DkfF7iezO0nJUPLcJK3pvA==", + "dependencies": { + "browserslist": "^4.21.4", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-custom-media": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/postcss-custom-media/-/postcss-custom-media-8.0.2.tgz", + "integrity": "sha512-7yi25vDAoHAkbhAzX9dHx2yc6ntS4jQvejrNcC+csQJAXjj15e7VcWfMgLqBNAbOvqi5uIa9huOVwdHbf+sKqg==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.3" + } + }, + "node_modules/postcss-custom-properties": { + "version": "12.1.11", + "resolved": "https://registry.npmjs.org/postcss-custom-properties/-/postcss-custom-properties-12.1.11.tgz", + "integrity": "sha512-0IDJYhgU8xDv1KY6+VgUwuQkVtmYzRwu+dMjnmdMafXYv86SWqfxkc7qdDvWS38vsjaEtv8e0vGOUQrAiMBLpQ==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-custom-selectors": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/postcss-custom-selectors/-/postcss-custom-selectors-6.0.3.tgz", + "integrity": "sha512-fgVkmyiWDwmD3JbpCmB45SvvlCD6z9CG6Ie6Iere22W5aHea6oWa7EM2bpnv2Fj3I94L3VbtvX9KqwSi5aFzSg==", + "dependencies": { + "postcss-selector-parser": "^6.0.4" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.3" + } + }, + "node_modules/postcss-dir-pseudo-class": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/postcss-dir-pseudo-class/-/postcss-dir-pseudo-class-6.0.5.tgz", + "integrity": "sha512-eqn4m70P031PF7ZQIvSgy9RSJ5uI2171O/OO/zcRNYpJbvaeKFUlar1aJ7rmgiQtbm0FSPsRewjpdS0Oew7MPA==", + "dependencies": { + "postcss-selector-parser": "^6.0.10" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-discard-comments": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-5.1.2.tgz", + "integrity": "sha512-+L8208OVbHVF2UQf1iDmRcbdjJkuBF6IS29yBDSiWUIzpYaAhtNl6JYnYm12FnkeCwQqF5LeklOu6rAqgfBZqQ==", + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-discard-duplicates": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-5.1.0.tgz", + "integrity": "sha512-zmX3IoSI2aoenxHV6C7plngHWWhUOV3sP1T8y2ifzxzbtnuhk1EdPwm0S1bIUNaJ2eNbWeGLEwzw8huPD67aQw==", + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-discard-empty": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-5.1.1.tgz", + "integrity": "sha512-zPz4WljiSuLWsI0ir4Mcnr4qQQ5e1Ukc3i7UfE2XcrwKK2LIPIqE5jxMRxO6GbI3cv//ztXDsXwEWT3BHOGh3A==", + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-discard-overridden": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-5.1.0.tgz", + "integrity": "sha512-21nOL7RqWR1kasIVdKs8HNqQJhFxLsyRfAnUDm4Fe4t4mCWL9OJiHvlHPjcd8zc5Myu89b/7wZDnOSjFgeWRtw==", + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-double-position-gradients": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/postcss-double-position-gradients/-/postcss-double-position-gradients-3.1.2.tgz", + "integrity": "sha512-GX+FuE/uBR6eskOK+4vkXgT6pDkexLokPaz/AbJna9s5Kzp/yl488pKPjhy0obB475ovfT1Wv8ho7U/cHNaRgQ==", + "dependencies": { + "@csstools/postcss-progressive-custom-properties": "^1.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-env-function": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/postcss-env-function/-/postcss-env-function-4.0.6.tgz", + "integrity": "sha512-kpA6FsLra+NqcFnL81TnsU+Z7orGtDTxcOhl6pwXeEq1yFPpRMkCDpHhrz8CFQDr/Wfm0jLiNQ1OsGGPjlqPwA==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/postcss-flexbugs-fixes": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-flexbugs-fixes/-/postcss-flexbugs-fixes-5.0.2.tgz", + "integrity": "sha512-18f9voByak7bTktR2QgDveglpn9DTbBWPUzSOe9g0N4WR/2eSt6Vrcbf0hmspvMI6YWGywz6B9f7jzpFNJJgnQ==", + "peerDependencies": { + "postcss": "^8.1.4" + } + }, + "node_modules/postcss-focus-visible": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/postcss-focus-visible/-/postcss-focus-visible-6.0.4.tgz", + "integrity": "sha512-QcKuUU/dgNsstIK6HELFRT5Y3lbrMLEOwG+A4s5cA+fx3A3y/JTq3X9LaOj3OC3ALH0XqyrgQIgey/MIZ8Wczw==", + "dependencies": { + "postcss-selector-parser": "^6.0.9" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/postcss-focus-within": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/postcss-focus-within/-/postcss-focus-within-5.0.4.tgz", + "integrity": "sha512-vvjDN++C0mu8jz4af5d52CB184ogg/sSxAFS+oUJQq2SuCe7T5U2iIsVJtsCp2d6R4j0jr5+q3rPkBVZkXD9fQ==", + "dependencies": { + "postcss-selector-parser": "^6.0.9" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/postcss-font-variant": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/postcss-font-variant/-/postcss-font-variant-5.0.0.tgz", + "integrity": "sha512-1fmkBaCALD72CK2a9i468mA/+tr9/1cBxRRMXOUaZqO43oWPR5imcyPjXwuv7PXbCid4ndlP5zWhidQVVa3hmA==", + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-gap-properties": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/postcss-gap-properties/-/postcss-gap-properties-3.0.5.tgz", + "integrity": "sha512-IuE6gKSdoUNcvkGIqdtjtcMtZIFyXZhmFd5RUlg97iVEvp1BZKV5ngsAjCjrVy+14uhGBQl9tzmi1Qwq4kqVOg==", + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-image-set-function": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/postcss-image-set-function/-/postcss-image-set-function-4.0.7.tgz", + "integrity": "sha512-9T2r9rsvYzm5ndsBE8WgtrMlIT7VbtTfE7b3BQnudUqnBcBo7L758oc+o+pdj/dUV0l5wjwSdjeOH2DZtfv8qw==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-import": { + "version": "15.1.0", + "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz", + "integrity": "sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==", + "dependencies": { + "postcss-value-parser": "^4.0.0", + "read-cache": "^1.0.0", + "resolve": "^1.1.7" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "postcss": "^8.0.0" + } + }, + "node_modules/postcss-initial": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-initial/-/postcss-initial-4.0.1.tgz", + "integrity": "sha512-0ueD7rPqX8Pn1xJIjay0AZeIuDoF+V+VvMt/uOnn+4ezUKhZM/NokDeP6DwMNyIoYByuN/94IQnt5FEkaN59xQ==", + "peerDependencies": { + "postcss": "^8.0.0" + } + }, + "node_modules/postcss-js": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-js/-/postcss-js-4.0.1.tgz", + "integrity": "sha512-dDLF8pEO191hJMtlHFPRa8xsizHaM82MLfNkUHdUtVEV3tgTp5oj+8qbEqYM57SLfc74KSbw//4SeJma2LRVIw==", + "dependencies": { + "camelcase-css": "^2.0.1" + }, + "engines": { + "node": "^12 || ^14 || >= 16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + "peerDependencies": { + "postcss": "^8.4.21" + } + }, + "node_modules/postcss-lab-function": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/postcss-lab-function/-/postcss-lab-function-4.2.1.tgz", + "integrity": "sha512-xuXll4isR03CrQsmxyz92LJB2xX9n+pZJ5jE9JgcnmsCammLyKdlzrBin+25dy6wIjfhJpKBAN80gsTlCgRk2w==", + "dependencies": { + "@csstools/postcss-progressive-custom-properties": "^1.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-load-config": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-4.0.1.tgz", + "integrity": "sha512-vEJIc8RdiBRu3oRAI0ymerOn+7rPuMvRXslTvZUKZonDHFIczxztIyJ1urxM1x9JXEikvpWWTUUqal5j/8QgvA==", + "dependencies": { + "lilconfig": "^2.0.5", + "yaml": "^2.1.1" + }, + "engines": { + "node": ">= 14" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + "peerDependencies": { + "postcss": ">=8.0.9", + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "postcss": { + "optional": true + }, + "ts-node": { + "optional": true + } + } + }, + "node_modules/postcss-load-config/node_modules/yaml": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.3.1.tgz", + "integrity": "sha512-2eHWfjaoXgTBC2jNM1LRef62VQa0umtvRiDSk6HSzW7RvS5YtkabJrwYLLEKWBc8a5U2PTSCs+dJjUTJdlHsWQ==", + "engines": { + "node": ">= 14" + } + }, + "node_modules/postcss-loader": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/postcss-loader/-/postcss-loader-6.2.1.tgz", + "integrity": "sha512-WbbYpmAaKcux/P66bZ40bpWsBucjx/TTgVVzRZ9yUO8yQfVBlameJ0ZGVaPfH64hNSBh63a+ICP5nqOpBA0w+Q==", + "dependencies": { + "cosmiconfig": "^7.0.0", + "klona": "^2.0.5", + "semver": "^7.3.5" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "postcss": "^7.0.0 || ^8.0.1", + "webpack": "^5.0.0" + } + }, + "node_modules/postcss-logical": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/postcss-logical/-/postcss-logical-5.0.4.tgz", + "integrity": "sha512-RHXxplCeLh9VjinvMrZONq7im4wjWGlRJAqmAVLXyZaXwfDWP73/oq4NdIp+OZwhQUMj0zjqDfM5Fj7qby+B4g==", + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.4" + } + }, + "node_modules/postcss-media-minmax": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/postcss-media-minmax/-/postcss-media-minmax-5.0.0.tgz", + "integrity": "sha512-yDUvFf9QdFZTuCUg0g0uNSHVlJ5X1lSzDZjPSFaiCWvjgsvu8vEVxtahPrLMinIDEEGnx6cBe6iqdx5YWz08wQ==", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-merge-longhand": { + "version": "5.1.7", + "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-5.1.7.tgz", + "integrity": "sha512-YCI9gZB+PLNskrK0BB3/2OzPnGhPkBEwmwhfYk1ilBHYVAZB7/tkTHFBAnCrvBBOmeYyMYw3DMjT55SyxMBzjQ==", + "dependencies": { + "postcss-value-parser": "^4.2.0", + "stylehacks": "^5.1.1" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-merge-rules": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-5.1.4.tgz", + "integrity": "sha512-0R2IuYpgU93y9lhVbO/OylTtKMVcHb67zjWIfCiKR9rWL3GUk1677LAqD/BcHizukdZEjT8Ru3oHRoAYoJy44g==", + "dependencies": { + "browserslist": "^4.21.4", + "caniuse-api": "^3.0.0", + "cssnano-utils": "^3.1.0", + "postcss-selector-parser": "^6.0.5" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-font-values": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-5.1.0.tgz", + "integrity": "sha512-el3mYTgx13ZAPPirSVsHqFzl+BBBDrXvbySvPGFnQcTI4iNslrPaFq4muTkLZmKlGk4gyFAYUBMH30+HurREyA==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-gradients": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-5.1.1.tgz", + "integrity": "sha512-VGvXMTpCEo4qHTNSa9A0a3D+dxGFZCYwR6Jokk+/3oB6flu2/PnPXAh2x7x52EkY5xlIHLm+Le8tJxe/7TNhzw==", + "dependencies": { + "colord": "^2.9.1", + "cssnano-utils": "^3.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-params": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-5.1.4.tgz", + "integrity": "sha512-+mePA3MgdmVmv6g+30rn57USjOGSAyuxUmkfiWpzalZ8aiBkdPYjXWtHuwJGm1v5Ojy0Z0LaSYhHaLJQB0P8Jw==", + "dependencies": { + "browserslist": "^4.21.4", + "cssnano-utils": "^3.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-selectors": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-5.2.1.tgz", + "integrity": "sha512-nPJu7OjZJTsVUmPdm2TcaiohIwxP+v8ha9NehQ2ye9szv4orirRU3SDdtUmKH+10nzn0bAyOXZ0UEr7OpvLehg==", + "dependencies": { + "postcss-selector-parser": "^6.0.5" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-modules-extract-imports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-3.0.0.tgz", + "integrity": "sha512-bdHleFnP3kZ4NYDhuGlVK+CMrQ/pqUm8bx/oGL93K6gVwiclvX5x0n76fYMKuIGKzlABOy13zsvqjb0f92TEXw==", + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-modules-local-by-default": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.0.3.tgz", + "integrity": "sha512-2/u2zraspoACtrbFRnTijMiQtb4GW4BvatjaG/bCjYQo8kLTdevCUlwuBHx2sCnSyrI3x3qj4ZK1j5LQBgzmwA==", + "dependencies": { + "icss-utils": "^5.0.0", + "postcss-selector-parser": "^6.0.2", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-modules-scope": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-3.0.0.tgz", + "integrity": "sha512-hncihwFA2yPath8oZ15PZqvWGkWf+XUfQgUGamS4LqoP1anQLOsOJw0vr7J7IwLpoY9fatA2qiGUGmuZL0Iqlg==", + "dependencies": { + "postcss-selector-parser": "^6.0.4" + }, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-modules-values": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-4.0.0.tgz", + "integrity": "sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ==", + "dependencies": { + "icss-utils": "^5.0.0" + }, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-nested": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/postcss-nested/-/postcss-nested-6.0.1.tgz", + "integrity": "sha512-mEp4xPMi5bSWiMbsgoPfcP74lsWLHkQbZc3sY+jWYd65CUwXrUaTp0fmNpa01ZcETKlIgUdFN/MpS2xZtqL9dQ==", + "dependencies": { + "postcss-selector-parser": "^6.0.11" + }, + "engines": { + "node": ">=12.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + "peerDependencies": { + "postcss": "^8.2.14" + } + }, + "node_modules/postcss-nesting": { + "version": "10.2.0", + "resolved": "https://registry.npmjs.org/postcss-nesting/-/postcss-nesting-10.2.0.tgz", + "integrity": "sha512-EwMkYchxiDiKUhlJGzWsD9b2zvq/r2SSubcRrgP+jujMXFzqvANLt16lJANC+5uZ6hjI7lpRmI6O8JIl+8l1KA==", + "dependencies": { + "@csstools/selector-specificity": "^2.0.0", + "postcss-selector-parser": "^6.0.10" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-normalize": { + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize/-/postcss-normalize-10.0.1.tgz", + "integrity": "sha512-+5w18/rDev5mqERcG3W5GZNMJa1eoYYNGo8gB7tEwaos0ajk3ZXAI4mHGcNT47NE+ZnZD1pEpUOFLvltIwmeJA==", + "dependencies": { + "@csstools/normalize.css": "*", + "postcss-browser-comments": "^4", + "sanitize.css": "*" + }, + "engines": { + "node": ">= 12" + }, + "peerDependencies": { + "browserslist": ">= 4", + "postcss": ">= 8" + } + }, + "node_modules/postcss-normalize-charset": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-5.1.0.tgz", + "integrity": "sha512-mSgUJ+pd/ldRGVx26p2wz9dNZ7ji6Pn8VWBajMXFf8jk7vUoSrZ2lt/wZR7DtlZYKesmZI680qjr2CeFF2fbUg==", + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-display-values": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-5.1.0.tgz", + "integrity": "sha512-WP4KIM4o2dazQXWmFaqMmcvsKmhdINFblgSeRgn8BJ6vxaMyaJkwAzpPpuvSIoG/rmX3M+IrRZEz2H0glrQNEA==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-positions": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-5.1.1.tgz", + "integrity": "sha512-6UpCb0G4eofTCQLFVuI3EVNZzBNPiIKcA1AKVka+31fTVySphr3VUgAIULBhxZkKgwLImhzMR2Bw1ORK+37INg==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-repeat-style": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-5.1.1.tgz", + "integrity": "sha512-mFpLspGWkQtBcWIRFLmewo8aC3ImN2i/J3v8YCFUwDnPu3Xz4rLohDO26lGjwNsQxB3YF0KKRwspGzE2JEuS0g==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-string": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-5.1.0.tgz", + "integrity": "sha512-oYiIJOf4T9T1N4i+abeIc7Vgm/xPCGih4bZz5Nm0/ARVJ7K6xrDlLwvwqOydvyL3RHNf8qZk6vo3aatiw/go3w==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-timing-functions": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-5.1.0.tgz", + "integrity": "sha512-DOEkzJ4SAXv5xkHl0Wa9cZLF3WCBhF3o1SKVxKQAa+0pYKlueTpCgvkFAHfk+Y64ezX9+nITGrDZeVGgITJXjg==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-unicode": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-5.1.1.tgz", + "integrity": "sha512-qnCL5jzkNUmKVhZoENp1mJiGNPcsJCs1aaRmURmeJGES23Z/ajaln+EPTD+rBeNkSryI+2WTdW+lwcVdOikrpA==", + "dependencies": { + "browserslist": "^4.21.4", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-url": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-5.1.0.tgz", + "integrity": "sha512-5upGeDO+PVthOxSmds43ZeMeZfKH+/DKgGRD7TElkkyS46JXAUhMzIKiCa7BabPeIy3AQcTkXwVVN7DbqsiCew==", + "dependencies": { + "normalize-url": "^6.0.1", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-whitespace": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-5.1.1.tgz", + "integrity": "sha512-83ZJ4t3NUDETIHTa3uEg6asWjSBYL5EdkVB0sDncx9ERzOKBVJIUeDO9RyA9Zwtig8El1d79HBp0JEi8wvGQnA==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-opacity-percentage": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/postcss-opacity-percentage/-/postcss-opacity-percentage-1.1.3.tgz", + "integrity": "sha512-An6Ba4pHBiDtyVpSLymUUERMo2cU7s+Obz6BTrS+gxkbnSBNKSuD0AVUc+CpBMrpVPKKfoVz0WQCX+Tnst0i4A==", + "funding": [ + { + "type": "kofi", + "url": "https://ko-fi.com/mrcgrtz" + }, + { + "type": "liberapay", + "url": "https://liberapay.com/mrcgrtz" + } + ], + "engines": { + "node": "^12 || ^14 || >=16" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-ordered-values": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-5.1.3.tgz", + "integrity": "sha512-9UO79VUhPwEkzbb3RNpqqghc6lcYej1aveQteWY+4POIwlqkYE21HKWaLDF6lWNuqCobEAyTovVhtI32Rbv2RQ==", + "dependencies": { + "cssnano-utils": "^3.1.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-overflow-shorthand": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/postcss-overflow-shorthand/-/postcss-overflow-shorthand-3.0.4.tgz", + "integrity": "sha512-otYl/ylHK8Y9bcBnPLo3foYFLL6a6Ak+3EQBPOTR7luMYCOsiVTUk1iLvNf6tVPNGXcoL9Hoz37kpfriRIFb4A==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-page-break": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/postcss-page-break/-/postcss-page-break-3.0.4.tgz", + "integrity": "sha512-1JGu8oCjVXLa9q9rFTo4MbeeA5FMe00/9C7lN4va606Rdb+HkxXtXsmEDrIraQ11fGz/WvKWa8gMuCKkrXpTsQ==", + "peerDependencies": { + "postcss": "^8" + } + }, + "node_modules/postcss-place": { + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/postcss-place/-/postcss-place-7.0.5.tgz", + "integrity": "sha512-wR8igaZROA6Z4pv0d+bvVrvGY4GVHihBCBQieXFY3kuSuMyOmEnnfFzHl/tQuqHZkfkIVBEbDvYcFfHmpSet9g==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-preset-env": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/postcss-preset-env/-/postcss-preset-env-7.8.3.tgz", + "integrity": "sha512-T1LgRm5uEVFSEF83vHZJV2z19lHg4yJuZ6gXZZkqVsqv63nlr6zabMH3l4Pc01FQCyfWVrh2GaUeCVy9Po+Aag==", + "dependencies": { + "@csstools/postcss-cascade-layers": "^1.1.1", + "@csstools/postcss-color-function": "^1.1.1", + "@csstools/postcss-font-format-keywords": "^1.0.1", + "@csstools/postcss-hwb-function": "^1.0.2", + "@csstools/postcss-ic-unit": "^1.0.1", + "@csstools/postcss-is-pseudo-class": "^2.0.7", + "@csstools/postcss-nested-calc": "^1.0.0", + "@csstools/postcss-normalize-display-values": "^1.0.1", + "@csstools/postcss-oklab-function": "^1.1.1", + "@csstools/postcss-progressive-custom-properties": "^1.3.0", + "@csstools/postcss-stepped-value-functions": "^1.0.1", + "@csstools/postcss-text-decoration-shorthand": "^1.0.0", + "@csstools/postcss-trigonometric-functions": "^1.0.2", + "@csstools/postcss-unset-value": "^1.0.2", + "autoprefixer": "^10.4.13", + "browserslist": "^4.21.4", + "css-blank-pseudo": "^3.0.3", + "css-has-pseudo": "^3.0.4", + "css-prefers-color-scheme": "^6.0.3", + "cssdb": "^7.1.0", + "postcss-attribute-case-insensitive": "^5.0.2", + "postcss-clamp": "^4.1.0", + "postcss-color-functional-notation": "^4.2.4", + "postcss-color-hex-alpha": "^8.0.4", + "postcss-color-rebeccapurple": "^7.1.1", + "postcss-custom-media": "^8.0.2", + "postcss-custom-properties": "^12.1.10", + "postcss-custom-selectors": "^6.0.3", + "postcss-dir-pseudo-class": "^6.0.5", + "postcss-double-position-gradients": "^3.1.2", + "postcss-env-function": "^4.0.6", + "postcss-focus-visible": "^6.0.4", + "postcss-focus-within": "^5.0.4", + "postcss-font-variant": "^5.0.0", + "postcss-gap-properties": "^3.0.5", + "postcss-image-set-function": "^4.0.7", + "postcss-initial": "^4.0.1", + "postcss-lab-function": "^4.2.1", + "postcss-logical": "^5.0.4", + "postcss-media-minmax": "^5.0.0", + "postcss-nesting": "^10.2.0", + "postcss-opacity-percentage": "^1.1.2", + "postcss-overflow-shorthand": "^3.0.4", + "postcss-page-break": "^3.0.4", + "postcss-place": "^7.0.5", + "postcss-pseudo-class-any-link": "^7.1.6", + "postcss-replace-overflow-wrap": "^4.0.0", + "postcss-selector-not": "^6.0.1", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-pseudo-class-any-link": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/postcss-pseudo-class-any-link/-/postcss-pseudo-class-any-link-7.1.6.tgz", + "integrity": "sha512-9sCtZkO6f/5ML9WcTLcIyV1yz9D1rf0tWc+ulKcvV30s0iZKS/ONyETvoWsr6vnrmW+X+KmuK3gV/w5EWnT37w==", + "dependencies": { + "postcss-selector-parser": "^6.0.10" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-reduce-initial": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-5.1.2.tgz", + "integrity": "sha512-dE/y2XRaqAi6OvjzD22pjTUQ8eOfc6m/natGHgKFBK9DxFmIm69YmaRVQrGgFlEfc1HePIurY0TmDeROK05rIg==", + "dependencies": { + "browserslist": "^4.21.4", + "caniuse-api": "^3.0.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-reduce-transforms": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-5.1.0.tgz", + "integrity": "sha512-2fbdbmgir5AvpW9RLtdONx1QoYG2/EtqpNQbFASDlixBbAYuTcJ0dECwlqNqH7VbaUnEnh8SrxOe2sRIn24XyQ==", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-replace-overflow-wrap": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-replace-overflow-wrap/-/postcss-replace-overflow-wrap-4.0.0.tgz", + "integrity": "sha512-KmF7SBPphT4gPPcKZc7aDkweHiKEEO8cla/GjcBK+ckKxiZslIu3C4GCRW3DNfL0o7yW7kMQu9xlZ1kXRXLXtw==", + "peerDependencies": { + "postcss": "^8.0.3" + } + }, + "node_modules/postcss-selector-not": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/postcss-selector-not/-/postcss-selector-not-6.0.1.tgz", + "integrity": "sha512-1i9affjAe9xu/y9uqWH+tD4r6/hDaXJruk8xn2x1vzxC2U3J3LKO3zJW4CyxlNhA56pADJ/djpEwpH1RClI2rQ==", + "dependencies": { + "postcss-selector-parser": "^6.0.10" + }, + "engines": { + "node": "^12 || ^14 || >=16" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + }, + "peerDependencies": { + "postcss": "^8.2" + } + }, + "node_modules/postcss-selector-parser": { + "version": "6.0.13", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.13.tgz", + "integrity": "sha512-EaV1Gl4mUEV4ddhDnv/xtj7sxwrwxdetHdWUGnT4VJQf+4d05v6lHYZr8N573k5Z0BViss7BDhfWtKS3+sfAqQ==", + "dependencies": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/postcss-svgo": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-5.1.0.tgz", + "integrity": "sha512-D75KsH1zm5ZrHyxPakAxJWtkyXew5qwS70v56exwvw542d9CRtTo78K0WeFxZB4G7JXKKMbEZtZayTGdIky/eA==", + "dependencies": { + "postcss-value-parser": "^4.2.0", + "svgo": "^2.7.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-svgo/node_modules/commander": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", + "engines": { + "node": ">= 10" + } + }, + "node_modules/postcss-svgo/node_modules/css-tree": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.1.3.tgz", + "integrity": "sha512-tRpdppF7TRazZrjJ6v3stzv93qxRcSsFmW6cX0Zm2NVKpxE1WV1HblnghVv9TreireHkqI/VDEsfolRF1p6y7Q==", + "dependencies": { + "mdn-data": "2.0.14", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/postcss-svgo/node_modules/mdn-data": { + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.14.tgz", + "integrity": "sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow==" + }, + "node_modules/postcss-svgo/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postcss-svgo/node_modules/svgo": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/svgo/-/svgo-2.8.0.tgz", + "integrity": "sha512-+N/Q9kV1+F+UeWYoSiULYo4xYSDQlTgb+ayMobAXPwMnLvop7oxKMo9OzIrX5x3eS4L4f2UHhc9axXwY8DpChg==", + "dependencies": { + "@trysound/sax": "0.2.0", + "commander": "^7.2.0", + "css-select": "^4.1.3", + "css-tree": "^1.1.3", + "csso": "^4.2.0", + "picocolors": "^1.0.0", + "stable": "^0.1.8" + }, + "bin": { + "svgo": "bin/svgo" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/postcss-unique-selectors": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-5.1.1.tgz", + "integrity": "sha512-5JiODlELrz8L2HwxfPnhOWZYWDxVHWL83ufOv84NrcgipI7TaeRsatAhK4Tr2/ZiYldpK/wBvw5BD3qfaK96GA==", + "dependencies": { + "postcss-selector-parser": "^6.0.5" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-value-parser": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", + "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==" + }, + "node_modules/postcss/node_modules/nanoid": { + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.6.tgz", + "integrity": "sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA==", + "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/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/pretty-bytes": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-5.6.0.tgz", + "integrity": "sha512-FFw039TmrBqFK8ma/7OL3sDz/VytdtJr044/QUJtH0wK9lb9jLq9tJyIxUwtQJHwar2BqtiA4iCWSwo9JLkzFg==", + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/pretty-error": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/pretty-error/-/pretty-error-4.0.0.tgz", + "integrity": "sha512-AoJ5YMAcXKYxKhuJGdcvse+Voc6v1RgnsR3nWcYU7q4t6z0Q6T86sv5Zq8VIRbOWWFpvdGE83LtdSMNd+6Y0xw==", + "dependencies": { + "lodash": "^4.17.20", + "renderkid": "^3.0.0" + } + }, + "node_modules/pretty-format": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "dependencies": { + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^17.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" + }, + "node_modules/promise": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/promise/-/promise-8.3.0.tgz", + "integrity": "sha512-rZPNPKTOYVNEEKFaq1HqTgOwZD+4/YHS5ukLzQCypkj+OkYx7iv0mA91lJlpPPZ8vMau3IIGj5Qlwrx+8iiSmg==", + "dependencies": { + "asap": "~2.0.6" + } + }, + "node_modules/prompts": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", + "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", + "dependencies": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/prop-types": { + "version": "15.8.1", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz", + "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==", + "dependencies": { + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.13.1" + } + }, + "node_modules/prop-types/node_modules/react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==" + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/proxy-addr/node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "engines": { + "node": ">= 0.10" + } + }, + "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/psl": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz", + "integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==" + }, + "node_modules/punycode": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz", + "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/q": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz", + "integrity": "sha512-kV/CThkXo6xyFEZUugw/+pIOywXcDbFYgSct5cT3gqlbkBE1SJdwy6UQoZvodiWF/ckQLZyDE/Bu1M6gVu5lVw==", + "engines": { + "node": ">=0.6.0", + "teleport": ">=0.2.0" + } + }, + "node_modules/qs": { + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz", + "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==", + "dependencies": { + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/querystringify": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz", + "integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==" + }, + "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==", + "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/raf": { + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/raf/-/raf-3.4.1.tgz", + "integrity": "sha512-Sq4CW4QhwOHE8ucn6J34MqtZCeWFP2aQSmrlroYgqAV1PjStIhJXxYuTgUIfkEk7zTLjmIjLmU5q+fbD1NnOJA==", + "dependencies": { + "performance-now": "^2.1.0" + } + }, + "node_modules/randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dependencies": { + "safe-buffer": "^5.1.0" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.5.1", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz", + "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==", + "dependencies": { + "bytes": "3.1.2", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/raw-body/node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/raw-body/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react/-/react-18.2.0.tgz", + "integrity": "sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==", + "dependencies": { + "loose-envify": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-app-polyfill": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/react-app-polyfill/-/react-app-polyfill-3.0.0.tgz", + "integrity": "sha512-sZ41cxiU5llIB003yxxQBYrARBqe0repqPTTYBTmMqTz9szeBbE37BehCE891NZsmdZqqP+xWKdT3eo3vOzN8w==", + "dependencies": { + "core-js": "^3.19.2", + "object-assign": "^4.1.1", + "promise": "^8.1.0", + "raf": "^3.4.1", + "regenerator-runtime": "^0.13.9", + "whatwg-fetch": "^3.6.2" + }, + "engines": { + "node": ">=14" + } + }, + "node_modules/react-app-polyfill/node_modules/regenerator-runtime": { + "version": "0.13.11", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz", + "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==" + }, + "node_modules/react-dev-utils": { + "version": "12.0.1", + "resolved": "https://registry.npmjs.org/react-dev-utils/-/react-dev-utils-12.0.1.tgz", + "integrity": "sha512-84Ivxmr17KjUupyqzFode6xKhjwuEJDROWKJy/BthkL7Wn6NJ8h4WE6k/exAv6ImS+0oZLRRW5j/aINMHyeGeQ==", + "dependencies": { + "@babel/code-frame": "^7.16.0", + "address": "^1.1.2", + "browserslist": "^4.18.1", + "chalk": "^4.1.2", + "cross-spawn": "^7.0.3", + "detect-port-alt": "^1.1.6", + "escape-string-regexp": "^4.0.0", + "filesize": "^8.0.6", + "find-up": "^5.0.0", + "fork-ts-checker-webpack-plugin": "^6.5.0", + "global-modules": "^2.0.0", + "globby": "^11.0.4", + "gzip-size": "^6.0.0", + "immer": "^9.0.7", + "is-root": "^2.1.0", + "loader-utils": "^3.2.0", + "open": "^8.4.0", + "pkg-up": "^3.1.0", + "prompts": "^2.4.2", + "react-error-overlay": "^6.0.11", + "recursive-readdir": "^2.2.2", + "shell-quote": "^1.7.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "engines": { + "node": ">=14" + } + }, + "node_modules/react-dev-utils/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/react-dev-utils/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/react-dev-utils/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/react-dev-utils/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==" + }, + "node_modules/react-dev-utils/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==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/react-dev-utils/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/react-dev-utils/node_modules/loader-utils": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-3.2.1.tgz", + "integrity": "sha512-ZvFw1KWS3GVyYBYb7qkmRM/WwL2TQQBxgCK62rlvm4WpVQ23Nb4tYjApUlfjrEGvOs7KHEsmyUn75OHZrJMWPw==", + "engines": { + "node": ">= 12.13.0" + } + }, + "node_modules/react-dev-utils/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/react-dom": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", + "integrity": "sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==", + "dependencies": { + "loose-envify": "^1.1.0", + "scheduler": "^0.23.0" + }, + "peerDependencies": { + "react": "^18.2.0" + } + }, + "node_modules/react-error-overlay": { + "version": "6.0.11", + "resolved": "https://registry.npmjs.org/react-error-overlay/-/react-error-overlay-6.0.11.tgz", + "integrity": "sha512-/6UZ2qgEyH2aqzYZgQPxEnz33NJ2gNsnHA2o5+o4wW9bLM/JYQitNP9xPhsXwC08hMMovfGe/8retsdDsczPRg==" + }, + "node_modules/react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==" + }, + "node_modules/react-refresh": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.11.0.tgz", + "integrity": "sha512-F27qZr8uUqwhWZboondsPx8tnC3Ct3SxZA3V5WyEvujRyyNv0VYPhoBg1gZ8/MV5tubQp76Trw8lTv9hzRBa+A==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-router": { + "version": "6.16.0", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-6.16.0.tgz", + "integrity": "sha512-VT4Mmc4jj5YyjpOi5jOf0I+TYzGpvzERy4ckNSvSh2RArv8LLoCxlsZ2D+tc7zgjxcY34oTz2hZaeX5RVprKqA==", + "dependencies": { + "@remix-run/router": "1.9.0" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "react": ">=16.8" + } + }, + "node_modules/react-router-dom": { + "version": "6.16.0", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.16.0.tgz", + "integrity": "sha512-aTfBLv3mk/gaKLxgRDUPbPw+s4Y/O+ma3rEN1u8EgEpLpPe6gNjIsWt9rxushMHHMb7mSwxRGdGlGdvmFsyPIg==", + "dependencies": { + "@remix-run/router": "1.9.0", + "react-router": "6.16.0" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "react": ">=16.8", + "react-dom": ">=16.8" + } + }, + "node_modules/react-scripts": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/react-scripts/-/react-scripts-5.0.1.tgz", + "integrity": "sha512-8VAmEm/ZAwQzJ+GOMLbBsTdDKOpuZh7RPs0UymvBR2vRk4iZWCskjbFnxqjrzoIvlNNRZ3QJFx6/qDSi6zSnaQ==", + "dependencies": { + "@babel/core": "^7.16.0", + "@pmmmwh/react-refresh-webpack-plugin": "^0.5.3", + "@svgr/webpack": "^5.5.0", + "babel-jest": "^27.4.2", + "babel-loader": "^8.2.3", + "babel-plugin-named-asset-import": "^0.3.8", + "babel-preset-react-app": "^10.0.1", + "bfj": "^7.0.2", + "browserslist": "^4.18.1", + "camelcase": "^6.2.1", + "case-sensitive-paths-webpack-plugin": "^2.4.0", + "css-loader": "^6.5.1", + "css-minimizer-webpack-plugin": "^3.2.0", + "dotenv": "^10.0.0", + "dotenv-expand": "^5.1.0", + "eslint": "^8.3.0", + "eslint-config-react-app": "^7.0.1", + "eslint-webpack-plugin": "^3.1.1", + "file-loader": "^6.2.0", + "fs-extra": "^10.0.0", + "html-webpack-plugin": "^5.5.0", + "identity-obj-proxy": "^3.0.0", + "jest": "^27.4.3", + "jest-resolve": "^27.4.2", + "jest-watch-typeahead": "^1.0.0", + "mini-css-extract-plugin": "^2.4.5", + "postcss": "^8.4.4", + "postcss-flexbugs-fixes": "^5.0.2", + "postcss-loader": "^6.2.1", + "postcss-normalize": "^10.0.1", + "postcss-preset-env": "^7.0.1", + "prompts": "^2.4.2", + "react-app-polyfill": "^3.0.0", + "react-dev-utils": "^12.0.1", + "react-refresh": "^0.11.0", + "resolve": "^1.20.0", + "resolve-url-loader": "^4.0.0", + "sass-loader": "^12.3.0", + "semver": "^7.3.5", + "source-map-loader": "^3.0.0", + "style-loader": "^3.3.1", + "tailwindcss": "^3.0.2", + "terser-webpack-plugin": "^5.2.5", + "webpack": "^5.64.4", + "webpack-dev-server": "^4.6.0", + "webpack-manifest-plugin": "^4.0.2", + "workbox-webpack-plugin": "^6.4.1" + }, + "bin": { + "react-scripts": "bin/react-scripts.js" + }, + "engines": { + "node": ">=14.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.2" + }, + "peerDependencies": { + "react": ">= 16", + "typescript": "^3.2.1 || ^4" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/read-cache": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz", + "integrity": "sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==", + "dependencies": { + "pify": "^2.3.0" + } + }, + "node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/recursive-readdir": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/recursive-readdir/-/recursive-readdir-2.2.3.tgz", + "integrity": "sha512-8HrF5ZsXk5FAH9dgsx3BlUer73nIhuj+9OrQwEbLTPOBzGkL1lsFCR01am+v+0m2Cmbs1nP12hLDl5FA7EszKA==", + "dependencies": { + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/redent": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/redent/-/redent-3.0.0.tgz", + "integrity": "sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==", + "dependencies": { + "indent-string": "^4.0.0", + "strip-indent": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/reflect.getprototypeof": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.3.tgz", + "integrity": "sha512-TTAOZpkJ2YLxl7mVHWrNo3iDMEkYlva/kgFcXndqMgbo/AZUmmavEkdXV+hXtE4P8xdyEKRzalaFqZVuwIk/Nw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "get-intrinsic": "^1.1.1", + "globalthis": "^1.0.3", + "which-builtin-type": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/regenerate": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.2.tgz", + "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==" + }, + "node_modules/regenerate-unicode-properties": { + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-10.1.0.tgz", + "integrity": "sha512-d1VudCLoIGitcU/hEg2QqvyGZQmdC0Lf8BqdOMXGFSvJP4bNV1+XqbPQeHHLD51Jh4QJJ225dlIFvY4Ly6MXmQ==", + "dependencies": { + "regenerate": "^1.4.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/regenerator-runtime": { + "version": "0.14.0", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.0.tgz", + "integrity": "sha512-srw17NI0TUWHuGa5CFGGmhfNIeja30WMBfbslPNhf6JrqQlLN5gcrvig1oqPxiVaXb0oW0XRKtH6Nngs5lKCIA==" + }, + "node_modules/regenerator-transform": { + "version": "0.15.2", + "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.15.2.tgz", + "integrity": "sha512-hfMp2BoF0qOk3uc5V20ALGDS2ddjQaLrdl7xrGXvAIow7qeWRM2VA2HuCHkUKk9slq3VwEwLNK3DFBqDfPGYtg==", + "dependencies": { + "@babel/runtime": "^7.8.4" + } + }, + "node_modules/regex-parser": { + "version": "2.2.11", + "resolved": "https://registry.npmjs.org/regex-parser/-/regex-parser-2.2.11.tgz", + "integrity": "sha512-jbD/FT0+9MBU2XAZluI7w2OBs1RBi6p9M83nkoZayQXXU9e8Robt69FcZc7wU4eJD/YFTjn1JdCk3rbMJajz8Q==" + }, + "node_modules/regexp.prototype.flags": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.0.tgz", + "integrity": "sha512-0SutC3pNudRKgquxGoRGIz946MZVHqbNfPjBdxeOhBrdgDKlRoXmYLQN9xRbrR09ZXWeGAdPuif7egofn6v5LA==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "functions-have-names": "^1.2.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/regexpu-core": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-5.3.2.tgz", + "integrity": "sha512-RAM5FlZz+Lhmo7db9L298p2vHP5ZywrVXmVXpmAD9GuL5MPH6t9ROw1iA/wfHkQ76Qe7AaPF0nGuim96/IrQMQ==", + "dependencies": { + "@babel/regjsgen": "^0.8.0", + "regenerate": "^1.4.2", + "regenerate-unicode-properties": "^10.1.0", + "regjsparser": "^0.9.1", + "unicode-match-property-ecmascript": "^2.0.0", + "unicode-match-property-value-ecmascript": "^2.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/regjsparser": { + "version": "0.9.1", + "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.9.1.tgz", + "integrity": "sha512-dQUtn90WanSNl+7mQKcXAgZxvUe7Z0SqXlgzv0za4LwiUhyzBC58yQO3liFoUgu8GiJVInAhJjkj1N0EtQ5nkQ==", + "dependencies": { + "jsesc": "~0.5.0" + }, + "bin": { + "regjsparser": "bin/parser" + } + }, + "node_modules/regjsparser/node_modules/jsesc": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", + "integrity": "sha512-uZz5UnB7u4T9LvwmFqXii7pZSouaRPorGs5who1Ip7VO0wxanFvBL7GkM6dTHlgX+jhBApRetaWpnDabOeTcnA==", + "bin": { + "jsesc": "bin/jsesc" + } + }, + "node_modules/relateurl": { + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/relateurl/-/relateurl-0.2.7.tgz", + "integrity": "sha512-G08Dxvm4iDN3MLM0EsP62EDV9IuhXPR6blNz6Utcp7zyV3tr4HVNINt6MpaRWbxoOHT3Q7YN2P+jaHX8vUbgog==", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/renderkid": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/renderkid/-/renderkid-3.0.0.tgz", + "integrity": "sha512-q/7VIQA8lmM1hF+jn+sFSPWGlMkSAeNYcPLmDQx2zzuiDfaLrOmumR8iaUKlenFgh0XRPIUeSPlH3A+AW3Z5pg==", + "dependencies": { + "css-select": "^4.1.3", + "dom-converter": "^0.2.0", + "htmlparser2": "^6.1.0", + "lodash": "^4.17.21", + "strip-ansi": "^6.0.1" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/requires-port": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", + "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==" + }, + "node_modules/resolve": { + "version": "1.22.4", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.4.tgz", + "integrity": "sha512-PXNdCiPqDqeUou+w1C2eTQbNfxKSuMxqTCuvlmmMsk1NWHL5fRrhY6Pl0qEYYc6+QqGClco1Qj8XnjPego4wfg==", + "dependencies": { + "is-core-module": "^2.13.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dependencies": { + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-url-loader": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-url-loader/-/resolve-url-loader-4.0.0.tgz", + "integrity": "sha512-05VEMczVREcbtT7Bz+C+96eUO5HDNvdthIiMB34t7FcF8ehcu4wC0sSgPUubs3XW2Q3CNLJk/BJrCU9wVRymiA==", + "dependencies": { + "adjust-sourcemap-loader": "^4.0.0", + "convert-source-map": "^1.7.0", + "loader-utils": "^2.0.0", + "postcss": "^7.0.35", + "source-map": "0.6.1" + }, + "engines": { + "node": ">=8.9" + }, + "peerDependencies": { + "rework": "1.0.1", + "rework-visit": "1.0.0" + }, + "peerDependenciesMeta": { + "rework": { + "optional": true + }, + "rework-visit": { + "optional": true + } + } + }, + "node_modules/resolve-url-loader/node_modules/picocolors": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-0.2.1.tgz", + "integrity": "sha512-cMlDqaLEqfSaW8Z7N5Jw+lyIW869EzT73/F5lhtY9cLGoVxSXznfgfXMO0Z5K0o0Q2TkTXq+0KFsdnSe3jDViA==" + }, + "node_modules/resolve-url-loader/node_modules/postcss": { + "version": "7.0.39", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.39.tgz", + "integrity": "sha512-yioayjNbHn6z1/Bywyb2Y4s3yvDAeXGOyxqD+LnVOinq6Mdmd++SW2wUNVzavyyHxd6+DxzWGIuosg6P1Rj8uA==", + "dependencies": { + "picocolors": "^0.2.1", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + } + }, + "node_modules/resolve-url-loader/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve.exports": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-1.1.1.tgz", + "integrity": "sha512-/NtpHNDN7jWhAaQ9BvBUYZ6YTXsRBgfqWFWP7BZBaoMJO/I3G5OFzvTuWNlZC3aPjins1F+TNrLKsGbH4rfsRQ==", + "engines": { + "node": ">=10" + } + }, + "node_modules/retry": { + "version": "0.13.1", + "resolved": "https://registry.npmjs.org/retry/-/retry-0.13.1.tgz", + "integrity": "sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==", + "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==", + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rollup": { + "version": "2.79.1", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.79.1.tgz", + "integrity": "sha512-uKxbd0IhMZOhjAiD5oAFp7BqvkA4Dv47qpOCtaNvng4HBwdbWtdOh8f5nZNuk2rp51PMGk3bzfWu5oayNEuYnw==", + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=10.0.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/rollup-plugin-terser": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/rollup-plugin-terser/-/rollup-plugin-terser-7.0.2.tgz", + "integrity": "sha512-w3iIaU4OxcF52UUXiZNsNeuXIMDvFrr+ZXK6bFZ0Q60qyVfq4uLptoS4bbq3paG3x216eQllFZX7zt6TIImguQ==", + "deprecated": "This package has been deprecated and is no longer maintained. Please use @rollup/plugin-terser", + "dependencies": { + "@babel/code-frame": "^7.10.4", + "jest-worker": "^26.2.1", + "serialize-javascript": "^4.0.0", + "terser": "^5.0.0" + }, + "peerDependencies": { + "rollup": "^2.0.0" + } + }, + "node_modules/rollup-plugin-terser/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/rollup-plugin-terser/node_modules/jest-worker": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.6.2.tgz", + "integrity": "sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==", + "dependencies": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^7.0.0" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/rollup-plugin-terser/node_modules/serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dependencies": { + "randombytes": "^2.1.0" + } + }, + "node_modules/rollup-plugin-terser/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "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==", + "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/safe-array-concat": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.0.0.tgz", + "integrity": "sha512-9dVEFruWIsnie89yym+xWTAYASdpw3CJV7Li/6zBewGf9z2i1j31rP6jnY0pHEO4QZh6N0K11bFjWmdR8UGdPQ==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.2.0", + "has-symbols": "^1.0.3", + "isarray": "^2.0.5" + }, + "engines": { + "node": ">=0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "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/safe-regex-test": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz", + "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.3", + "is-regex": "^1.1.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/sanitize.css": { + "version": "13.0.0", + "resolved": "https://registry.npmjs.org/sanitize.css/-/sanitize.css-13.0.0.tgz", + "integrity": "sha512-ZRwKbh/eQ6w9vmTjkuG0Ioi3HBwPFce0O+v//ve+aOq1oeCy7jMV2qzzAlpsNuqpqCBjjriM1lbtZbF/Q8jVyA==" + }, + "node_modules/sass-loader": { + "version": "12.6.0", + "resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-12.6.0.tgz", + "integrity": "sha512-oLTaH0YCtX4cfnJZxKSLAyglED0naiYfNG1iXfU5w1LNZ+ukoA5DtyDIN5zmKVZwYNJP4KRc5Y3hkWga+7tYfA==", + "dependencies": { + "klona": "^2.0.4", + "neo-async": "^2.6.2" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "fibers": ">= 3.1.0", + "node-sass": "^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0", + "sass": "^1.3.0", + "sass-embedded": "*", + "webpack": "^5.0.0" + }, + "peerDependenciesMeta": { + "fibers": { + "optional": true + }, + "node-sass": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + } + } + }, + "node_modules/sax": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", + "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" + }, + "node_modules/saxes": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz", + "integrity": "sha512-5LBh1Tls8c9xgGjw3QrMwETmTMVk0oFgvrFSvWx62llR2hcEInrKNZ2GZCCuuy2lvWrdl5jhbpeqc5hRYKFOcw==", + "dependencies": { + "xmlchars": "^2.2.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/scheduler": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.0.tgz", + "integrity": "sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==", + "dependencies": { + "loose-envify": "^1.1.0" + } + }, + "node_modules/schema-utils": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.3.0.tgz", + "integrity": "sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==", + "dependencies": { + "@types/json-schema": "^7.0.8", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/select-hose": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/select-hose/-/select-hose-2.0.0.tgz", + "integrity": "sha512-mEugaLK+YfkijB4fx0e6kImuJdCIt2LxCRcbEYPqRGCs4F2ogyfZU5IAZRdjCP8JPq2AtdNoC/Dux63d9Kiryg==" + }, + "node_modules/selfsigned": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/selfsigned/-/selfsigned-2.1.1.tgz", + "integrity": "sha512-GSL3aowiF7wa/WtSFwnUrludWFoNhftq8bUkH9pkzjpN2XSPOAYEgg6e0sS9s0rZwgJzJiQRPU18A6clnoW5wQ==", + "dependencies": { + "node-forge": "^1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/semver": { + "version": "7.5.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", + "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/semver/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/semver/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" + }, + "node_modules/send": { + "version": "0.18.0", + "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz", + "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", + "dependencies": { + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "mime": "1.6.0", + "ms": "2.1.3", + "on-finished": "2.4.1", + "range-parser": "~1.2.1", + "statuses": "2.0.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/send/node_modules/debug/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/send/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==" + }, + "node_modules/serialize-javascript": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.1.tgz", + "integrity": "sha512-owoXEFjWRllis8/M1Q+Cw5k8ZH40e3zhp/ovX+Xr/vi1qj6QesbyXXViFbpNvWvPNAD62SutwEXavefrLJWj7w==", + "dependencies": { + "randombytes": "^2.1.0" + } + }, + "node_modules/serve-index": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/serve-index/-/serve-index-1.9.1.tgz", + "integrity": "sha512-pXHfKNP4qujrtteMrSBb0rc8HJ9Ms/GrXwcUtUtD5s4ewDJI8bT3Cz2zTVRMKtri49pLx2e0Ya8ziP5Ya2pZZw==", + "dependencies": { + "accepts": "~1.3.4", + "batch": "0.6.1", + "debug": "2.6.9", + "escape-html": "~1.0.3", + "http-errors": "~1.6.2", + "mime-types": "~2.1.17", + "parseurl": "~1.3.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/serve-index/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/serve-index/node_modules/depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/serve-index/node_modules/http-errors": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha512-lks+lVC8dgGyh97jxvxeYTWQFvh4uw4yC12gVl63Cg30sjPX4wuGcdkICVXDAESr6OJGjqGA8Iz5mkeN6zlD7A==", + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/serve-index/node_modules/inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==" + }, + "node_modules/serve-index/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/serve-index/node_modules/setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==" + }, + "node_modules/serve-index/node_modules/statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/serve-static": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", + "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", + "dependencies": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.18.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" + }, + "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==", + "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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/shell-quote": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/shell-quote/-/shell-quote-1.8.1.tgz", + "integrity": "sha512-6j1W9l1iAs/4xYBI1SYOVZyFcCis9b4KCLQ8fgAGG07QvzaRLVVRQvAy85yNmmZSjYjg4MWh4gNvlPujU/5LpA==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", + "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", + "dependencies": { + "call-bind": "^1.0.0", + "get-intrinsic": "^1.0.2", + "object-inspect": "^1.9.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==" + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==" + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "engines": { + "node": ">=8" + } + }, + "node_modules/sockjs": { + "version": "0.3.24", + "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.24.tgz", + "integrity": "sha512-GJgLTZ7vYb/JtPSSZ10hsOYIvEYsjbNU+zPdIHcUaWVNUEPivzxku31865sSSud0Da0W4lEeOPlmw93zLQchuQ==", + "dependencies": { + "faye-websocket": "^0.11.3", + "uuid": "^8.3.2", + "websocket-driver": "^0.7.4" + } + }, + "node_modules/source-list-map": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/source-list-map/-/source-list-map-2.0.1.tgz", + "integrity": "sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==" + }, + "node_modules/source-map": { + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.4.tgz", + "integrity": "sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==", + "engines": { + "node": ">= 8" + } + }, + "node_modules/source-map-js": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", + "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-loader": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/source-map-loader/-/source-map-loader-3.0.2.tgz", + "integrity": "sha512-BokxPoLjyl3iOrgkWaakaxqnelAJSS+0V+De0kKIq6lyWrXuiPgYTGp6z3iHmqljKAaLXwZa+ctD8GccRJeVvg==", + "dependencies": { + "abab": "^2.0.5", + "iconv-lite": "^0.6.3", + "source-map-js": "^1.0.1" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.0.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.21", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", + "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/source-map-support/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sourcemap-codec": { + "version": "1.4.8", + "resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz", + "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==", + "deprecated": "Please use @jridgewell/sourcemap-codec instead" + }, + "node_modules/spdy": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/spdy/-/spdy-4.0.2.tgz", + "integrity": "sha512-r46gZQZQV+Kl9oItvl1JZZqJKGr+oEkB08A6BzkiR7593/7IbtuncXHd2YoYeTsG4157ZssMu9KYvUHLcjcDoA==", + "dependencies": { + "debug": "^4.1.0", + "handle-thing": "^2.0.0", + "http-deceiver": "^1.2.7", + "select-hose": "^2.0.0", + "spdy-transport": "^3.0.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/spdy-transport": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/spdy-transport/-/spdy-transport-3.0.0.tgz", + "integrity": "sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw==", + "dependencies": { + "debug": "^4.1.0", + "detect-node": "^2.0.4", + "hpack.js": "^2.1.6", + "obuf": "^1.1.2", + "readable-stream": "^3.0.6", + "wbuf": "^1.7.3" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==" + }, + "node_modules/stable": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz", + "integrity": "sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w==", + "deprecated": "Modern JS already guarantees Array#sort() is a stable sort, so this library is deprecated. See the compatibility table on MDN: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#browser_compatibility" + }, + "node_modules/stack-utils": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.6.tgz", + "integrity": "sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==", + "dependencies": { + "escape-string-regexp": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/stack-utils/node_modules/escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "engines": { + "node": ">=8" + } + }, + "node_modules/stackframe": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/stackframe/-/stackframe-1.3.4.tgz", + "integrity": "sha512-oeVtt7eWQS+Na6F//S4kJ2K2VbRlS9D43mAlMyVpVWovy9o+jfgH8O9agzANzaiLjclA0oYzUXEM4PurhSUChw==" + }, + "node_modules/statuses": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", + "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/stop-iteration-iterator": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.0.0.tgz", + "integrity": "sha512-iCGQj+0l0HOdZ2AEeBADlsRC+vsnDsZsbdSiH1yNSjcfKM7fdpCMfqAL/dwF5BLiw/XhRft/Wax6zQbhq2BcjQ==", + "dependencies": { + "internal-slot": "^1.0.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, + "node_modules/string-length": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", + "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/string-natural-compare": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/string-natural-compare/-/string-natural-compare-3.0.1.tgz", + "integrity": "sha512-n3sPwynL1nwKi3WJ6AIsClwBMa0zTi54fn2oLU6ndfTSIO05xaznjSf15PcBZU6FNWbmN5Q6cxT4V5hGvB4taw==" + }, + "node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "node_modules/string.prototype.matchall": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.8.tgz", + "integrity": "sha512-6zOCOcJ+RJAQshcTvXPHoxoQGONa3e/Lqx90wUA+wEzX78sg5Bo+1tQo4N0pohS0erG9qtCqJDjNCQBjeWVxyg==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4", + "get-intrinsic": "^1.1.3", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.3", + "regexp.prototype.flags": "^1.4.3", + "side-channel": "^1.0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trim": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.7.tgz", + "integrity": "sha512-p6TmeT1T3411M8Cgg9wBTMRtY2q9+PNy9EV1i2lIXUN/btt763oIfxwN3RR8VU6wHX8j/1CFy0L+YuThm6bgOg==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimend": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz", + "integrity": "sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimstart": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz", + "integrity": "sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.20.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/stringify-object": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/stringify-object/-/stringify-object-3.3.0.tgz", + "integrity": "sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw==", + "dependencies": { + "get-own-enumerable-property-symbols": "^3.0.0", + "is-obj": "^1.0.1", + "is-regexp": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-comments/-/strip-comments-2.0.1.tgz", + "integrity": "sha512-ZprKx+bBLXv067WTCALv8SSz5l2+XhpYCsVtSqlMnkAXMWDq+/ekVbl1ghqP9rUHTzv6sm/DwCOiYutU/yp1fw==", + "engines": { + "node": ">=10" + } + }, + "node_modules/strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-indent": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-3.0.0.tgz", + "integrity": "sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==", + "dependencies": { + "min-indent": "^1.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "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==", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/style-loader": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/style-loader/-/style-loader-3.3.3.tgz", + "integrity": "sha512-53BiGLXAcll9maCYtZi2RCQZKa8NQQai5C4horqKyRmHj9H7QmcUyucrH+4KW/gBQbXM2AsB0axoEcFZPlfPcw==", + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.0.0" + } + }, + "node_modules/stylehacks": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-5.1.1.tgz", + "integrity": "sha512-sBpcd5Hx7G6seo7b1LkpttvTz7ikD0LlH5RmdcBNb6fFR0Fl7LQwHDFr300q4cwUqi+IYrFGmsIHieMBfnN/Bw==", + "dependencies": { + "browserslist": "^4.21.4", + "postcss-selector-parser": "^6.0.4" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/sucrase": { + "version": "3.34.0", + "resolved": "https://registry.npmjs.org/sucrase/-/sucrase-3.34.0.tgz", + "integrity": "sha512-70/LQEZ07TEcxiU2dz51FKaE6hCTWC6vr7FOk3Gr0U60C3shtAN+H+BFr9XlYe5xqf3RA8nrc+VIwzCfnxuXJw==", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.2", + "commander": "^4.0.0", + "glob": "7.1.6", + "lines-and-columns": "^1.1.6", + "mz": "^2.7.0", + "pirates": "^4.0.1", + "ts-interface-checker": "^0.1.9" + }, + "bin": { + "sucrase": "bin/sucrase", + "sucrase-node": "bin/sucrase-node" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/sucrase/node_modules/commander": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", + "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/sucrase/node_modules/glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/supports-hyperlinks": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.3.0.tgz", + "integrity": "sha512-RpsAZlpWcDwOPQA22aCH4J0t7L8JmAvsCxfOSEwm7cQs3LshN36QaTkwd70DnBOXDWGssw2eUoc8CaRWT0XunA==", + "dependencies": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-hyperlinks/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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-hyperlinks/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==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/svg-parser": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/svg-parser/-/svg-parser-2.0.4.tgz", + "integrity": "sha512-e4hG1hRwoOdRb37cIMSgzNsxyzKfayW6VOflrwvR+/bzrkyxY/31WkbgnQpgtrNp1SdpJvpUAGTa/ZoiPNDuRQ==" + }, + "node_modules/svgo": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/svgo/-/svgo-1.3.2.tgz", + "integrity": "sha512-yhy/sQYxR5BkC98CY7o31VGsg014AKLEPxdfhora76l36hD9Rdy5NZA/Ocn6yayNPgSamYdtX2rFJdcv07AYVw==", + "deprecated": "This SVGO version is no longer supported. Upgrade to v2.x.x.", + "dependencies": { + "chalk": "^2.4.1", + "coa": "^2.0.2", + "css-select": "^2.0.0", + "css-select-base-adapter": "^0.1.1", + "css-tree": "1.0.0-alpha.37", + "csso": "^4.0.2", + "js-yaml": "^3.13.1", + "mkdirp": "~0.5.1", + "object.values": "^1.1.0", + "sax": "~1.2.4", + "stable": "^0.1.8", + "unquote": "~1.1.1", + "util.promisify": "~1.0.0" + }, + "bin": { + "svgo": "bin/svgo" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/svgo/node_modules/css-select": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-2.1.0.tgz", + "integrity": "sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ==", + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^3.2.1", + "domutils": "^1.7.0", + "nth-check": "^1.0.2" + } + }, + "node_modules/svgo/node_modules/css-what": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/css-what/-/css-what-3.4.2.tgz", + "integrity": "sha512-ACUm3L0/jiZTqfzRM3Hi9Q8eZqd6IK37mMWPLz9PJxkLWllYeRf+EHUSHYEtFop2Eqytaq1FizFVh7XfBnXCDQ==", + "engines": { + "node": ">= 6" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/svgo/node_modules/dom-serializer": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.2.2.tgz", + "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", + "dependencies": { + "domelementtype": "^2.0.1", + "entities": "^2.0.0" + } + }, + "node_modules/svgo/node_modules/domutils": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz", + "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", + "dependencies": { + "dom-serializer": "0", + "domelementtype": "1" + } + }, + "node_modules/svgo/node_modules/domutils/node_modules/domelementtype": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.1.tgz", + "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==" + }, + "node_modules/svgo/node_modules/nth-check": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-1.0.2.tgz", + "integrity": "sha512-WeBOdju8SnzPN5vTUJYxYUxLeXpCaVP5i5e0LF8fg7WORF2Wd7wFX/pk0tYZk7s8T+J7VLy0Da6J1+wCT0AtHg==", + "dependencies": { + "boolbase": "~1.0.0" + } + }, + "node_modules/symbol-tree": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", + "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==" + }, + "node_modules/tailwindcss": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.3.3.tgz", + "integrity": "sha512-A0KgSkef7eE4Mf+nKJ83i75TMyq8HqY3qmFIJSWy8bNt0v1lG7jUcpGpoTFxAwYcWOphcTBLPPJg+bDfhDf52w==", + "dependencies": { + "@alloc/quick-lru": "^5.2.0", + "arg": "^5.0.2", + "chokidar": "^3.5.3", + "didyoumean": "^1.2.2", + "dlv": "^1.1.3", + "fast-glob": "^3.2.12", + "glob-parent": "^6.0.2", + "is-glob": "^4.0.3", + "jiti": "^1.18.2", + "lilconfig": "^2.1.0", + "micromatch": "^4.0.5", + "normalize-path": "^3.0.0", + "object-hash": "^3.0.0", + "picocolors": "^1.0.0", + "postcss": "^8.4.23", + "postcss-import": "^15.1.0", + "postcss-js": "^4.0.1", + "postcss-load-config": "^4.0.1", + "postcss-nested": "^6.0.1", + "postcss-selector-parser": "^6.0.11", + "resolve": "^1.22.2", + "sucrase": "^3.32.0" + }, + "bin": { + "tailwind": "lib/cli.js", + "tailwindcss": "lib/cli.js" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/tapable": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.1.tgz", + "integrity": "sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/temp-dir": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/temp-dir/-/temp-dir-2.0.0.tgz", + "integrity": "sha512-aoBAniQmmwtcKp/7BzsH8Cxzv8OL736p7v1ihGb5e9DJ9kTwGWHrQrVB5+lfVDzfGrdRzXch+ig7LHaY1JTOrg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/tempy": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/tempy/-/tempy-0.6.0.tgz", + "integrity": "sha512-G13vtMYPT/J8A4X2SjdtBTphZlrp1gKv6hZiOjw14RCWg6GbHuQBGtjlx75xLbYV/wEc0D7G5K4rxKP/cXk8Bw==", + "dependencies": { + "is-stream": "^2.0.0", + "temp-dir": "^2.0.0", + "type-fest": "^0.16.0", + "unique-string": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/tempy/node_modules/type-fest": { + "version": "0.16.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.16.0.tgz", + "integrity": "sha512-eaBzG6MxNzEn9kiwvtre90cXaNLkmadMWa1zQMs3XORCXNbsH/OewwbxC5ia9dCxIxnTAsSxXJaa/p5y8DlvJg==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/terminal-link": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/terminal-link/-/terminal-link-2.1.1.tgz", + "integrity": "sha512-un0FmiRUQNr5PJqy9kP7c40F5BOfpGlYTrxonDChEZB7pzZxRNp/bt+ymiy9/npwXya9KH99nJ/GXFIiUkYGFQ==", + "dependencies": { + "ansi-escapes": "^4.2.1", + "supports-hyperlinks": "^2.0.0" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/terser": { + "version": "5.19.2", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.19.2.tgz", + "integrity": "sha512-qC5+dmecKJA4cpYxRa5aVkKehYsQKc+AHeKl0Oe62aYjBL8ZA33tTljktDHJSaxxMnbI5ZYw+o/S2DxxLu8OfA==", + "dependencies": { + "@jridgewell/source-map": "^0.3.3", + "acorn": "^8.8.2", + "commander": "^2.20.0", + "source-map-support": "~0.5.20" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/terser-webpack-plugin": { + "version": "5.3.9", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.9.tgz", + "integrity": "sha512-ZuXsqE07EcggTWQjXUj+Aot/OMcD0bMKGgF63f7UxYcu5/AJF53aIpK1YoP5xR9l6s/Hy2b+t1AM0bLNPRuhwA==", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.17", + "jest-worker": "^27.4.5", + "schema-utils": "^3.1.1", + "serialize-javascript": "^6.0.1", + "terser": "^5.16.8" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.1.0" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "esbuild": { + "optional": true + }, + "uglify-js": { + "optional": true + } + } + }, + "node_modules/terser/node_modules/commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" + }, + "node_modules/test-exclude": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", + "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", + "dependencies": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "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==" + }, + "node_modules/thenify": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz", + "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==", + "dependencies": { + "any-promise": "^1.0.0" + } + }, + "node_modules/thenify-all": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz", + "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==", + "dependencies": { + "thenify": ">= 3.1.0 < 4" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/throat": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/throat/-/throat-6.0.2.tgz", + "integrity": "sha512-WKexMoJj3vEuK0yFEapj8y64V0A6xcuPuK9Gt1d0R+dzCSJc0lHqQytAbSB4cDAK0dWh4T0E2ETkoLE2WZ41OQ==" + }, + "node_modules/thunky": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/thunky/-/thunky-1.1.0.tgz", + "integrity": "sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA==" + }, + "node_modules/tmpl": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", + "integrity": "sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==" + }, + "node_modules/to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "engines": { + "node": ">=4" + } + }, + "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==", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/tough-cookie": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.1.3.tgz", + "integrity": "sha512-aX/y5pVRkfRnfmuX+OdbSdXvPe6ieKX/G2s7e98f4poJHnqH3281gDPm/metm6E/WRamfx7WC4HUqkWHfQHprw==", + "dependencies": { + "psl": "^1.1.33", + "punycode": "^2.1.1", + "universalify": "^0.2.0", + "url-parse": "^1.5.3" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/tough-cookie/node_modules/universalify": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.2.0.tgz", + "integrity": "sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==", + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/tr46": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-2.1.0.tgz", + "integrity": "sha512-15Ih7phfcdP5YxqiB+iDtLoaTz4Nd35+IiAv0kQ5FNKHzXgdWqPoTIqEDDJmXceQt4JZk6lVPT8lnDlPpGDppw==", + "dependencies": { + "punycode": "^2.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tryer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/tryer/-/tryer-1.0.1.tgz", + "integrity": "sha512-c3zayb8/kWWpycWYg87P71E1S1ZL6b6IJxfb5fvsUgsf0S2MVGaDhDXXjDMpdCpfWXqptc+4mXwmiy1ypXqRAA==" + }, + "node_modules/ts-interface-checker": { + "version": "0.1.13", + "resolved": "https://registry.npmjs.org/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz", + "integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==" + }, + "node_modules/tsconfig-paths": { + "version": "3.14.2", + "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.14.2.tgz", + "integrity": "sha512-o/9iXgCYc5L/JxCHPe3Hvh8Q/2xm5Z+p18PESBU6Ff33695QnCHBEjcytY2q19ua7Mbl/DavtBOLq+oG0RCL+g==", + "dependencies": { + "@types/json5": "^0.0.29", + "json5": "^1.0.2", + "minimist": "^1.2.6", + "strip-bom": "^3.0.0" + } + }, + "node_modules/tsconfig-paths/node_modules/json5": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", + "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", + "dependencies": { + "minimist": "^1.2.0" + }, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/tsconfig-paths/node_modules/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", + "engines": { + "node": ">=4" + } + }, + "node_modules/tslib": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==" + }, + "node_modules/tsutils": { + "version": "3.21.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz", + "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", + "dependencies": { + "tslib": "^1.8.1" + }, + "engines": { + "node": ">= 6" + }, + "peerDependencies": { + "typescript": ">=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta" + } + }, + "node_modules/tsutils/node_modules/tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" + }, + "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==", + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-detect": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", + "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", + "engines": { + "node": ">=4" + } + }, + "node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typed-array-buffer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.0.tgz", + "integrity": "sha512-Y8KTSIglk9OZEr8zywiIHG/kmQ7KWyjseXs1CbSo8vC42w7hg2HgYTxSWwP0+is7bWDc1H+Fo026CpHFwm8tkw==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.2.1", + "is-typed-array": "^1.1.10" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/typed-array-byte-length": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.0.tgz", + "integrity": "sha512-Or/+kvLxNpeQ9DtSydonMxCx+9ZXOswtwJn17SNLvhptaXYDJvkFFP5zbfU/uLmvnBJlI4yrnXRxpdWH/M5tNA==", + "dependencies": { + "call-bind": "^1.0.2", + "for-each": "^0.3.3", + "has-proto": "^1.0.1", + "is-typed-array": "^1.1.10" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-byte-offset": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.0.tgz", + "integrity": "sha512-RD97prjEt9EL8YgAgpOkf3O4IF9lhJFr9g0htQkm0rchFp/Vx7LW5Q8fSXXub7BXAODyUQohRMyOc3faCPd0hg==", + "dependencies": { + "available-typed-arrays": "^1.0.5", + "call-bind": "^1.0.2", + "for-each": "^0.3.3", + "has-proto": "^1.0.1", + "is-typed-array": "^1.1.10" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-length": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz", + "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==", + "dependencies": { + "call-bind": "^1.0.2", + "for-each": "^0.3.3", + "is-typed-array": "^1.1.9" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typedarray-to-buffer": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", + "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", + "dependencies": { + "is-typedarray": "^1.0.0" + } + }, + "node_modules/typescript": { + "version": "4.9.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.5.tgz", + "integrity": "sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==", + "peer": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=4.2.0" + } + }, + "node_modules/unbox-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", + "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==", + "dependencies": { + "call-bind": "^1.0.2", + "has-bigints": "^1.0.2", + "has-symbols": "^1.0.3", + "which-boxed-primitive": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/unicode-canonical-property-names-ecmascript": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz", + "integrity": "sha512-yY5PpDlfVIU5+y/BSCxAJRBIS1Zc2dDG3Ujq+sR0U+JjUevW2JhocOF+soROYDSaAezOzOKuyyixhD6mBknSmQ==", + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-match-property-ecmascript": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-2.0.0.tgz", + "integrity": "sha512-5kaZCrbp5mmbz5ulBkDkbY0SsPOjKqVS35VpL9ulMPfSl0J0Xsm+9Evphv9CoIZFwre7aJoa94AY6seMKGVN5Q==", + "dependencies": { + "unicode-canonical-property-names-ecmascript": "^2.0.0", + "unicode-property-aliases-ecmascript": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-match-property-value-ecmascript": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-2.1.0.tgz", + "integrity": "sha512-qxkjQt6qjg/mYscYMC0XKRn3Rh0wFPlfxB0xkt9CfyTvpX1Ra0+rAmdX2QyAobptSEvuy4RtpPRui6XkV+8wjA==", + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-property-aliases-ecmascript": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-2.1.0.tgz", + "integrity": "sha512-6t3foTQI9qne+OZoVQB/8x8rk2k1eVy1gRXhV3oFQ5T6R1dqQ1xtin3XqSlx3+ATBkliTaR/hHyJBm+LVPNM8w==", + "engines": { + "node": ">=4" + } + }, + "node_modules/unique-string": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", + "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", + "dependencies": { + "crypto-random-string": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/universal-cookie": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/universal-cookie/-/universal-cookie-6.1.1.tgz", + "integrity": "sha512-33S9x3CpdUnnjwTNs2Fgc41WGve2tdLtvaK2kPSbZRc5pGpz2vQFbRWMxlATsxNNe/Cy8SzmnmbuBM85jpZPtA==", + "dependencies": { + "@types/cookie": "^0.5.1", + "cookie": "^0.5.0" + } + }, + "node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/unquote": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/unquote/-/unquote-1.1.1.tgz", + "integrity": "sha512-vRCqFv6UhXpWxZPyGDh/F3ZpNv8/qo7w6iufLpQg9aKnQ71qM4B5KiI7Mia9COcjEhrO9LueHpMYjYzsWH3OIg==" + }, + "node_modules/upath": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", + "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", + "engines": { + "node": ">=4", + "yarn": "*" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.11.tgz", + "integrity": "sha512-dCwEFf0/oT85M1fHBg4F0jtLwJrutGoHSQXCh7u4o2t1drG+c0a9Flnqww6XUKSfQMPpJBRjU8d4RXB09qtvaA==", + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "escalade": "^3.1.1", + "picocolors": "^1.0.0" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/url-parse": { + "version": "1.5.10", + "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz", + "integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==", + "dependencies": { + "querystringify": "^2.1.1", + "requires-port": "^1.0.0" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + }, + "node_modules/util.promisify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/util.promisify/-/util.promisify-1.0.1.tgz", + "integrity": "sha512-g9JpC/3He3bm38zsLupWryXHoEcS22YHthuPQSJdMy6KNrzIRzWqcsHzD/WUnqe45whVou4VIsPew37DoXWNrA==", + "dependencies": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.2", + "has-symbols": "^1.0.1", + "object.getownpropertydescriptors": "^2.1.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/utila": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/utila/-/utila-0.4.0.tgz", + "integrity": "sha512-Z0DbgELS9/L/75wZbro8xAnT50pBVFQZ+hUEueGDU5FN51YSCYM+jdxsfCiHjwNP/4LCDD0i/graKpeBnOXKRA==" + }, + "node_modules/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/uuid": { + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", + "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==", + "bin": { + "uuid": "dist/bin/uuid" + } + }, + "node_modules/v8-to-istanbul": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-8.1.1.tgz", + "integrity": "sha512-FGtKtv3xIpR6BYhvgH8MI/y78oT7d8Au3ww4QIxymrCtZEh5b8gCw2siywE+puhEmuWKDtmfrvF5UlB298ut3w==", + "dependencies": { + "@types/istanbul-lib-coverage": "^2.0.1", + "convert-source-map": "^1.6.0", + "source-map": "^0.7.3" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/w3c-hr-time": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/w3c-hr-time/-/w3c-hr-time-1.0.2.tgz", + "integrity": "sha512-z8P5DvDNjKDoFIHK7q8r8lackT6l+jo/Ye3HOle7l9nICP9lf1Ci25fy9vHd0JOWewkIFzXIEig3TdKT7JQ5fQ==", + "deprecated": "Use your platform's native performance.now() and performance.timeOrigin.", + "dependencies": { + "browser-process-hrtime": "^1.0.0" + } + }, + "node_modules/w3c-xmlserializer": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-2.0.0.tgz", + "integrity": "sha512-4tzD0mF8iSiMiNs30BiLO3EpfGLZUT2MSX/G+o7ZywDzliWQ3OPtTZ0PTC3B3ca1UAf4cJMHB+2Bf56EriJuRA==", + "dependencies": { + "xml-name-validator": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/walker": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.8.tgz", + "integrity": "sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==", + "dependencies": { + "makeerror": "1.0.12" + } + }, + "node_modules/watchpack": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.4.0.tgz", + "integrity": "sha512-Lcvm7MGST/4fup+ifyKi2hjyIAwcdI4HRgtvTpIUxBRhB+RFtUh8XtDOxUfctVCnhVi+QQj49i91OyvzkJl6cg==", + "dependencies": { + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.1.2" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/wbuf": { + "version": "1.7.3", + "resolved": "https://registry.npmjs.org/wbuf/-/wbuf-1.7.3.tgz", + "integrity": "sha512-O84QOnr0icsbFGLS0O3bI5FswxzRr8/gHwWkDlQFskhSPryQXvrTMxjxGP4+iWYoauLoBvfDpkrOauZ+0iZpDA==", + "dependencies": { + "minimalistic-assert": "^1.0.0" + } + }, + "node_modules/web-vitals": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/web-vitals/-/web-vitals-2.1.4.tgz", + "integrity": "sha512-sVWcwhU5mX6crfI5Vd2dC4qchyTqxV8URinzt25XqVh+bHEPGH4C3NPrNionCP7Obx59wrYEbNlw4Z8sjALzZg==" + }, + "node_modules/webidl-conversions": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-6.1.0.tgz", + "integrity": "sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w==", + "engines": { + "node": ">=10.4" + } + }, + "node_modules/webpack": { + "version": "5.88.2", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.88.2.tgz", + "integrity": "sha512-JmcgNZ1iKj+aiR0OvTYtWQqJwq37Pf683dY9bVORwVbUrDhLhdn/PlO2sHsFHPkj7sHNQF3JwaAkp49V+Sq1tQ==", + "dependencies": { + "@types/eslint-scope": "^3.7.3", + "@types/estree": "^1.0.0", + "@webassemblyjs/ast": "^1.11.5", + "@webassemblyjs/wasm-edit": "^1.11.5", + "@webassemblyjs/wasm-parser": "^1.11.5", + "acorn": "^8.7.1", + "acorn-import-assertions": "^1.9.0", + "browserslist": "^4.14.5", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^5.15.0", + "es-module-lexer": "^1.2.1", + "eslint-scope": "5.1.1", + "events": "^3.2.0", + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.2.9", + "json-parse-even-better-errors": "^2.3.1", + "loader-runner": "^4.2.0", + "mime-types": "^2.1.27", + "neo-async": "^2.6.2", + "schema-utils": "^3.2.0", + "tapable": "^2.1.1", + "terser-webpack-plugin": "^5.3.7", + "watchpack": "^2.4.0", + "webpack-sources": "^3.2.3" + }, + "bin": { + "webpack": "bin/webpack.js" + }, + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + } + } + }, + "node_modules/webpack-dev-middleware": { + "version": "5.3.3", + "resolved": "https://registry.npmjs.org/webpack-dev-middleware/-/webpack-dev-middleware-5.3.3.tgz", + "integrity": "sha512-hj5CYrY0bZLB+eTO+x/j67Pkrquiy7kWepMHmUMoPsmcUaeEnQJqFzHJOyxgWlq746/wUuA64p9ta34Kyb01pA==", + "dependencies": { + "colorette": "^2.0.10", + "memfs": "^3.4.3", + "mime-types": "^2.1.31", + "range-parser": "^1.2.1", + "schema-utils": "^4.0.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/webpack-dev-middleware/node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/webpack-dev-middleware/node_modules/ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dependencies": { + "fast-deep-equal": "^3.1.3" + }, + "peerDependencies": { + "ajv": "^8.8.2" + } + }, + "node_modules/webpack-dev-middleware/node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/webpack-dev-middleware/node_modules/schema-utils": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.2.0.tgz", + "integrity": "sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/webpack-dev-server": { + "version": "4.15.1", + "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-4.15.1.tgz", + "integrity": "sha512-5hbAst3h3C3L8w6W4P96L5vaV0PxSmJhxZvWKYIdgxOQm8pNZ5dEOmmSLBVpP85ReeyRt6AS1QJNyo/oFFPeVA==", + "dependencies": { + "@types/bonjour": "^3.5.9", + "@types/connect-history-api-fallback": "^1.3.5", + "@types/express": "^4.17.13", + "@types/serve-index": "^1.9.1", + "@types/serve-static": "^1.13.10", + "@types/sockjs": "^0.3.33", + "@types/ws": "^8.5.5", + "ansi-html-community": "^0.0.8", + "bonjour-service": "^1.0.11", + "chokidar": "^3.5.3", + "colorette": "^2.0.10", + "compression": "^1.7.4", + "connect-history-api-fallback": "^2.0.0", + "default-gateway": "^6.0.3", + "express": "^4.17.3", + "graceful-fs": "^4.2.6", + "html-entities": "^2.3.2", + "http-proxy-middleware": "^2.0.3", + "ipaddr.js": "^2.0.1", + "launch-editor": "^2.6.0", + "open": "^8.0.9", + "p-retry": "^4.5.0", + "rimraf": "^3.0.2", + "schema-utils": "^4.0.0", + "selfsigned": "^2.1.1", + "serve-index": "^1.9.1", + "sockjs": "^0.3.24", + "spdy": "^4.0.2", + "webpack-dev-middleware": "^5.3.1", + "ws": "^8.13.0" + }, + "bin": { + "webpack-dev-server": "bin/webpack-dev-server.js" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.37.0 || ^5.0.0" + }, + "peerDependenciesMeta": { + "webpack": { + "optional": true + }, + "webpack-cli": { + "optional": true + } + } + }, + "node_modules/webpack-dev-server/node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/webpack-dev-server/node_modules/ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dependencies": { + "fast-deep-equal": "^3.1.3" + }, + "peerDependencies": { + "ajv": "^8.8.2" + } + }, + "node_modules/webpack-dev-server/node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/webpack-dev-server/node_modules/schema-utils": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.2.0.tgz", + "integrity": "sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 12.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/webpack-dev-server/node_modules/ws": { + "version": "8.13.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.13.0.tgz", + "integrity": "sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA==", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/webpack-manifest-plugin": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/webpack-manifest-plugin/-/webpack-manifest-plugin-4.1.1.tgz", + "integrity": "sha512-YXUAwxtfKIJIKkhg03MKuiFAD72PlrqCiwdwO4VEXdRO5V0ORCNwaOwAZawPZalCbmH9kBDmXnNeQOw+BIEiow==", + "dependencies": { + "tapable": "^2.0.0", + "webpack-sources": "^2.2.0" + }, + "engines": { + "node": ">=12.22.0" + }, + "peerDependencies": { + "webpack": "^4.44.2 || ^5.47.0" + } + }, + "node_modules/webpack-manifest-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-manifest-plugin/node_modules/webpack-sources": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-2.3.1.tgz", + "integrity": "sha512-y9EI9AO42JjEcrTJFOYmVywVZdKVUfOvDUPsJea5GIr1JOEGFVqwlY2K098fFoIjOkDzHn2AjRvM8dsBZu+gCA==", + "dependencies": { + "source-list-map": "^2.0.1", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/webpack-sources": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-3.2.3.tgz", + "integrity": "sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==", + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/webpack/node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/webpack/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/websocket-driver": { + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.7.4.tgz", + "integrity": "sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==", + "dependencies": { + "http-parser-js": ">=0.5.1", + "safe-buffer": ">=5.1.0", + "websocket-extensions": ">=0.1.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/websocket-extensions": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/websocket-extensions/-/websocket-extensions-0.1.4.tgz", + "integrity": "sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg==", + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/whatwg-encoding": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.5.tgz", + "integrity": "sha512-b5lim54JOPN9HtzvK9HFXvBma/rnfFeqsic0hSpjtDbVxR3dJKLc+KB4V6GgiGOvl7CY/KNh8rxSo9DKQrnUEw==", + "dependencies": { + "iconv-lite": "0.4.24" + } + }, + "node_modules/whatwg-encoding/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/whatwg-fetch": { + "version": "3.6.17", + "resolved": "https://registry.npmjs.org/whatwg-fetch/-/whatwg-fetch-3.6.17.tgz", + "integrity": "sha512-c4ghIvG6th0eudYwKZY5keb81wtFz9/WeAHAoy8+r18kcWlitUIrmGFQ2rWEl4UCKUilD3zCLHOIPheHx5ypRQ==" + }, + "node_modules/whatwg-mimetype": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz", + "integrity": "sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g==" + }, + "node_modules/whatwg-url": { + "version": "8.7.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-8.7.0.tgz", + "integrity": "sha512-gAojqb/m9Q8a5IV96E3fHJM70AzCkgt4uXYX2O7EmuyOnLrViCQlsEBmF9UQIu3/aeAIp2U17rtbpZWNntQqdg==", + "dependencies": { + "lodash": "^4.7.0", + "tr46": "^2.1.0", + "webidl-conversions": "^6.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/which-boxed-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz", + "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==", + "dependencies": { + "is-bigint": "^1.0.1", + "is-boolean-object": "^1.1.0", + "is-number-object": "^1.0.4", + "is-string": "^1.0.5", + "is-symbol": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-builtin-type": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.1.3.tgz", + "integrity": "sha512-YmjsSMDBYsM1CaFiayOVT06+KJeXf0o5M/CAd4o1lTadFAtacTUM49zoYxr/oroopFDfhvN6iEcBxUyc3gvKmw==", + "dependencies": { + "function.prototype.name": "^1.1.5", + "has-tostringtag": "^1.0.0", + "is-async-function": "^2.0.0", + "is-date-object": "^1.0.5", + "is-finalizationregistry": "^1.0.2", + "is-generator-function": "^1.0.10", + "is-regex": "^1.1.4", + "is-weakref": "^1.0.2", + "isarray": "^2.0.5", + "which-boxed-primitive": "^1.0.2", + "which-collection": "^1.0.1", + "which-typed-array": "^1.1.9" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-collection": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.1.tgz", + "integrity": "sha512-W8xeTUwaln8i3K/cY1nGXzdnVZlidBcagyNFtBdD5kxnb4TvGKR7FfSIS3mYpwWS1QUCutfKz8IY8RjftB0+1A==", + "dependencies": { + "is-map": "^2.0.1", + "is-set": "^2.0.1", + "is-weakmap": "^2.0.1", + "is-weakset": "^2.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-typed-array": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.11.tgz", + "integrity": "sha512-qe9UWWpkeG5yzZ0tNYxDmd7vo58HDBc39mZ0xWWpolAGADdFOzkfamWLDxkOWcvHQKVmdTyQdLD4NOfjLWTKew==", + "dependencies": { + "available-typed-arrays": "^1.0.5", + "call-bind": "^1.0.2", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/workbox-background-sync": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-background-sync/-/workbox-background-sync-6.6.0.tgz", + "integrity": "sha512-jkf4ZdgOJxC9u2vztxLuPT/UjlH7m/nWRQ/MgGL0v8BJHoZdVGJd18Kck+a0e55wGXdqyHO+4IQTk0685g4MUw==", + "dependencies": { + "idb": "^7.0.1", + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-broadcast-update": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-broadcast-update/-/workbox-broadcast-update-6.6.0.tgz", + "integrity": "sha512-nm+v6QmrIFaB/yokJmQ/93qIJ7n72NICxIwQwe5xsZiV2aI93MGGyEyzOzDPVz5THEr5rC3FJSsO3346cId64Q==", + "dependencies": { + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-build": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-build/-/workbox-build-6.6.0.tgz", + "integrity": "sha512-Tjf+gBwOTuGyZwMz2Nk/B13Fuyeo0Q84W++bebbVsfr9iLkDSo6j6PST8tET9HYA58mlRXwlMGpyWO8ETJiXdQ==", + "dependencies": { + "@apideck/better-ajv-errors": "^0.3.1", + "@babel/core": "^7.11.1", + "@babel/preset-env": "^7.11.0", + "@babel/runtime": "^7.11.2", + "@rollup/plugin-babel": "^5.2.0", + "@rollup/plugin-node-resolve": "^11.2.1", + "@rollup/plugin-replace": "^2.4.1", + "@surma/rollup-plugin-off-main-thread": "^2.2.3", + "ajv": "^8.6.0", + "common-tags": "^1.8.0", + "fast-json-stable-stringify": "^2.1.0", + "fs-extra": "^9.0.1", + "glob": "^7.1.6", + "lodash": "^4.17.20", + "pretty-bytes": "^5.3.0", + "rollup": "^2.43.1", + "rollup-plugin-terser": "^7.0.0", + "source-map": "^0.8.0-beta.0", + "stringify-object": "^3.3.0", + "strip-comments": "^2.0.1", + "tempy": "^0.6.0", + "upath": "^1.2.0", + "workbox-background-sync": "6.6.0", + "workbox-broadcast-update": "6.6.0", + "workbox-cacheable-response": "6.6.0", + "workbox-core": "6.6.0", + "workbox-expiration": "6.6.0", + "workbox-google-analytics": "6.6.0", + "workbox-navigation-preload": "6.6.0", + "workbox-precaching": "6.6.0", + "workbox-range-requests": "6.6.0", + "workbox-recipes": "6.6.0", + "workbox-routing": "6.6.0", + "workbox-strategies": "6.6.0", + "workbox-streams": "6.6.0", + "workbox-sw": "6.6.0", + "workbox-window": "6.6.0" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/workbox-build/node_modules/@apideck/better-ajv-errors": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/@apideck/better-ajv-errors/-/better-ajv-errors-0.3.6.tgz", + "integrity": "sha512-P+ZygBLZtkp0qqOAJJVX4oX/sFo5JR3eBWwwuqHHhK0GIgQOKWrAfiAaWX0aArHkRWHMuggFEgAZNxVPwPZYaA==", + "dependencies": { + "json-schema": "^0.4.0", + "jsonpointer": "^5.0.0", + "leven": "^3.1.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "ajv": ">=8" + } + }, + "node_modules/workbox-build/node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/workbox-build/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/workbox-build/node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/workbox-build/node_modules/source-map": { + "version": "0.8.0-beta.0", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.8.0-beta.0.tgz", + "integrity": "sha512-2ymg6oRBpebeZi9UUNsgQ89bhx01TcTkmNTGnNO88imTmbSgy4nfujrgVEFKWpMTEGA11EDkTt7mqObTPdigIA==", + "dependencies": { + "whatwg-url": "^7.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/workbox-build/node_modules/tr46": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-1.0.1.tgz", + "integrity": "sha512-dTpowEjclQ7Kgx5SdBkqRzVhERQXov8/l9Ft9dVM9fmg0W0KQSVaXX9T4i6twCPNtYiZM53lpSSUAwJbFPOHxA==", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/workbox-build/node_modules/webidl-conversions": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-4.0.2.tgz", + "integrity": "sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==" + }, + "node_modules/workbox-build/node_modules/whatwg-url": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-7.1.0.tgz", + "integrity": "sha512-WUu7Rg1DroM7oQvGWfOiAK21n74Gg+T4elXEQYkOhtyLeWiJFoOGLXPKI/9gzIie9CtwVLm8wtw6YJdKyxSjeg==", + "dependencies": { + "lodash.sortby": "^4.7.0", + "tr46": "^1.0.1", + "webidl-conversions": "^4.0.2" + } + }, + "node_modules/workbox-cacheable-response": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-cacheable-response/-/workbox-cacheable-response-6.6.0.tgz", + "integrity": "sha512-JfhJUSQDwsF1Xv3EV1vWzSsCOZn4mQ38bWEBR3LdvOxSPgB65gAM6cS2CX8rkkKHRgiLrN7Wxoyu+TuH67kHrw==", + "deprecated": "workbox-background-sync@6.6.0", + "dependencies": { + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-core": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-core/-/workbox-core-6.6.0.tgz", + "integrity": "sha512-GDtFRF7Yg3DD859PMbPAYPeJyg5gJYXuBQAC+wyrWuuXgpfoOrIQIvFRZnQ7+czTIQjIr1DhLEGFzZanAT/3bQ==" + }, + "node_modules/workbox-expiration": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-expiration/-/workbox-expiration-6.6.0.tgz", + "integrity": "sha512-baplYXcDHbe8vAo7GYvyAmlS4f6998Jff513L4XvlzAOxcl8F620O91guoJ5EOf5qeXG4cGdNZHkkVAPouFCpw==", + "dependencies": { + "idb": "^7.0.1", + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-google-analytics": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-google-analytics/-/workbox-google-analytics-6.6.0.tgz", + "integrity": "sha512-p4DJa6OldXWd6M9zRl0H6vB9lkrmqYFkRQ2xEiNdBFp9U0LhsGO7hsBscVEyH9H2/3eZZt8c97NB2FD9U2NJ+Q==", + "dependencies": { + "workbox-background-sync": "6.6.0", + "workbox-core": "6.6.0", + "workbox-routing": "6.6.0", + "workbox-strategies": "6.6.0" + } + }, + "node_modules/workbox-navigation-preload": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-navigation-preload/-/workbox-navigation-preload-6.6.0.tgz", + "integrity": "sha512-utNEWG+uOfXdaZmvhshrh7KzhDu/1iMHyQOV6Aqup8Mm78D286ugu5k9MFD9SzBT5TcwgwSORVvInaXWbvKz9Q==", + "dependencies": { + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-precaching": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-precaching/-/workbox-precaching-6.6.0.tgz", + "integrity": "sha512-eYu/7MqtRZN1IDttl/UQcSZFkHP7dnvr/X3Vn6Iw6OsPMruQHiVjjomDFCNtd8k2RdjLs0xiz9nq+t3YVBcWPw==", + "dependencies": { + "workbox-core": "6.6.0", + "workbox-routing": "6.6.0", + "workbox-strategies": "6.6.0" + } + }, + "node_modules/workbox-range-requests": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-range-requests/-/workbox-range-requests-6.6.0.tgz", + "integrity": "sha512-V3aICz5fLGq5DpSYEU8LxeXvsT//mRWzKrfBOIxzIdQnV/Wj7R+LyJVTczi4CQ4NwKhAaBVaSujI1cEjXW+hTw==", + "dependencies": { + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-recipes": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-recipes/-/workbox-recipes-6.6.0.tgz", + "integrity": "sha512-TFi3kTgYw73t5tg73yPVqQC8QQjxJSeqjXRO4ouE/CeypmP2O/xqmB/ZFBBQazLTPxILUQ0b8aeh0IuxVn9a6A==", + "dependencies": { + "workbox-cacheable-response": "6.6.0", + "workbox-core": "6.6.0", + "workbox-expiration": "6.6.0", + "workbox-precaching": "6.6.0", + "workbox-routing": "6.6.0", + "workbox-strategies": "6.6.0" + } + }, + "node_modules/workbox-routing": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-routing/-/workbox-routing-6.6.0.tgz", + "integrity": "sha512-x8gdN7VDBiLC03izAZRfU+WKUXJnbqt6PG9Uh0XuPRzJPpZGLKce/FkOX95dWHRpOHWLEq8RXzjW0O+POSkKvw==", + "dependencies": { + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-strategies": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-strategies/-/workbox-strategies-6.6.0.tgz", + "integrity": "sha512-eC07XGuINAKUWDnZeIPdRdVja4JQtTuc35TZ8SwMb1ztjp7Ddq2CJ4yqLvWzFWGlYI7CG/YGqaETntTxBGdKgQ==", + "dependencies": { + "workbox-core": "6.6.0" + } + }, + "node_modules/workbox-streams": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-streams/-/workbox-streams-6.6.0.tgz", + "integrity": "sha512-rfMJLVvwuED09CnH1RnIep7L9+mj4ufkTyDPVaXPKlhi9+0czCu+SJggWCIFbPpJaAZmp2iyVGLqS3RUmY3fxg==", + "dependencies": { + "workbox-core": "6.6.0", + "workbox-routing": "6.6.0" + } + }, + "node_modules/workbox-sw": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-sw/-/workbox-sw-6.6.0.tgz", + "integrity": "sha512-R2IkwDokbtHUE4Kus8pKO5+VkPHD2oqTgl+XJwh4zbF1HyjAbgNmK/FneZHVU7p03XUt9ICfuGDYISWG9qV/CQ==" + }, + "node_modules/workbox-webpack-plugin": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-webpack-plugin/-/workbox-webpack-plugin-6.6.0.tgz", + "integrity": "sha512-xNZIZHalboZU66Wa7x1YkjIqEy1gTR+zPM+kjrYJzqN7iurYZBctBLISyScjhkJKYuRrZUP0iqViZTh8rS0+3A==", + "dependencies": { + "fast-json-stable-stringify": "^2.1.0", + "pretty-bytes": "^5.4.1", + "upath": "^1.2.0", + "webpack-sources": "^1.4.3", + "workbox-build": "6.6.0" + }, + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "webpack": "^4.4.0 || ^5.9.0" + } + }, + "node_modules/workbox-webpack-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/workbox-webpack-plugin/node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "node_modules/workbox-window": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/workbox-window/-/workbox-window-6.6.0.tgz", + "integrity": "sha512-L4N9+vka17d16geaJXXRjENLFldvkWy7JyGxElRD0JvBxvFEd8LOhr+uXCcar/NzAmIBRv9EZ+M+Qr4mOoBITw==", + "dependencies": { + "@types/trusted-types": "^2.0.2", + "workbox-core": "6.6.0" + } + }, + "node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi/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==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/wrap-ansi/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==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/wrap-ansi/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==" + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==" + }, + "node_modules/write-file-atomic": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", + "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", + "dependencies": { + "imurmurhash": "^0.1.4", + "is-typedarray": "^1.0.0", + "signal-exit": "^3.0.2", + "typedarray-to-buffer": "^3.1.5" + } + }, + "node_modules/ws": { + "version": "7.5.9", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.9.tgz", + "integrity": "sha512-F+P9Jil7UiSKSkppIiD94dN07AwvFixvLIj1Og1Rl9GGMuNipJnV9JzjD6XuqmAeiswGvUmNLjr5cFuXwNS77Q==", + "engines": { + "node": ">=8.3.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": "^5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/xml-name-validator": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", + "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==" + }, + "node_modules/xmlchars": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", + "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==" + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "engines": { + "node": ">=10" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==" + }, + "node_modules/yaml": { + "version": "1.10.2", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", + "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/yargs": { + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", + "dependencies": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/yargs-parser": { + "version": "20.2.9", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", + "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", + "engines": { + "node": ">=10" + } + }, + "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==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + } +} diff --git a/front-end/package.json b/front-end/package.json new file mode 100644 index 0000000..ec18ca2 --- /dev/null +++ b/front-end/package.json @@ -0,0 +1,47 @@ +{ + "name": "louk-chatwalker", + "version": "0.1.0", + "private": true, + "dependencies": { + "@fortawesome/free-solid-svg-icons": "^6.4.2", + "@fortawesome/react-fontawesome": "^0.2.0", + "@radix-ui/react-icons": "^1.3.2", + "@react-buddy/ide-toolbox": "^2.4.0", + "@testing-library/jest-dom": "^5.17.0", + "@testing-library/react": "^13.4.0", + "@testing-library/user-event": "^13.5.0", + "axios": "latest", + "jwt-decode": "latest", + "nanoid": "^5.0.1", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "react-router-dom": "^6.16.0", + "react-scripts": "5.0.1", + "universal-cookie": "^6.1.1", + "web-vitals": "^2.1.4" + }, + "scripts": { + "start": "react-scripts start", + "build": "react-scripts build", + "test": "react-scripts test", + "eject": "react-scripts eject" + }, + "eslintConfig": { + "extends": [ + "react-app", + "react-app/jest" + ] + }, + "browserslist": { + "production": [ + ">0.2%", + "not dead", + "not op_mini all" + ], + "development": [ + "last 1 chrome version", + "last 1 firefox version", + "last 1 safari version" + ] + } +} diff --git a/front-end/public/favicon.ico b/front-end/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..a11777cc471a4344702741ab1c8a588998b1311a GIT binary patch literal 3870 zcma);c{J4h9>;%nil|2-o+rCuEF-(I%-F}ijC~o(k~HKAkr0)!FCj~d>`RtpD?8b; zXOC1OD!V*IsqUwzbMF1)-gEDD=A573Z-&G7^LoAC9|WO7Xc0Cx1g^Zu0u_SjAPB3vGa^W|sj)80f#V0@M_CAZTIO(t--xg= z!sii`1giyH7EKL_+Wi0ab<)&E_0KD!3Rp2^HNB*K2@PHCs4PWSA32*-^7d{9nH2_E zmC{C*N*)(vEF1_aMamw2A{ZH5aIDqiabnFdJ|y0%aS|64E$`s2ccV~3lR!u<){eS` z#^Mx6o(iP1Ix%4dv`t@!&Za-K@mTm#vadc{0aWDV*_%EiGK7qMC_(`exc>-$Gb9~W!w_^{*pYRm~G zBN{nA;cm^w$VWg1O^^<6vY`1XCD|s_zv*g*5&V#wv&s#h$xlUilPe4U@I&UXZbL z0)%9Uj&@yd03n;!7do+bfixH^FeZ-Ema}s;DQX2gY+7g0s(9;`8GyvPY1*vxiF&|w z>!vA~GA<~JUqH}d;DfBSi^IT*#lrzXl$fNpq0_T1tA+`A$1?(gLb?e#0>UELvljtQ zK+*74m0jn&)5yk8mLBv;=@}c{t0ztT<v;Avck$S6D`Z)^c0(jiwKhQsn|LDRY&w(Fmi91I7H6S;b0XM{e zXp0~(T@k_r-!jkLwd1_Vre^v$G4|kh4}=Gi?$AaJ)3I+^m|Zyj#*?Kp@w(lQdJZf4 z#|IJW5z+S^e9@(6hW6N~{pj8|NO*>1)E=%?nNUAkmv~OY&ZV;m-%?pQ_11)hAr0oAwILrlsGawpxx4D43J&K=n+p3WLnlDsQ$b(9+4 z?mO^hmV^F8MV{4Lx>(Q=aHhQ1){0d*(e&s%G=i5rq3;t{JC zmgbn5Nkl)t@fPH$v;af26lyhH!k+#}_&aBK4baYPbZy$5aFx4}ka&qxl z$=Rh$W;U)>-=S-0=?7FH9dUAd2(q#4TCAHky!$^~;Dz^j|8_wuKc*YzfdAht@Q&ror?91Dm!N03=4=O!a)I*0q~p0g$Fm$pmr$ zb;wD;STDIi$@M%y1>p&_>%?UP($15gou_ue1u0!4(%81;qcIW8NyxFEvXpiJ|H4wz z*mFT(qVx1FKufG11hByuX%lPk4t#WZ{>8ka2efjY`~;AL6vWyQKpJun2nRiZYDij$ zP>4jQXPaP$UC$yIVgGa)jDV;F0l^n(V=HMRB5)20V7&r$jmk{UUIe zVjKroK}JAbD>B`2cwNQ&GDLx8{pg`7hbA~grk|W6LgiZ`8y`{Iq0i>t!3p2}MS6S+ zO_ruKyAElt)rdS>CtF7j{&6rP-#c=7evGMt7B6`7HG|-(WL`bDUAjyn+k$mx$CH;q2Dz4x;cPP$hW=`pFfLO)!jaCL@V2+F)So3}vg|%O*^T1j>C2lx zsURO-zIJC$^$g2byVbRIo^w>UxK}74^TqUiRR#7s_X$e)$6iYG1(PcW7un-va-S&u zHk9-6Zn&>T==A)lM^D~bk{&rFzCi35>UR!ZjQkdSiNX*-;l4z9j*7|q`TBl~Au`5& z+c)*8?#-tgUR$Zd%Q3bs96w6k7q@#tUn`5rj+r@_sAVVLqco|6O{ILX&U-&-cbVa3 zY?ngHR@%l{;`ri%H*0EhBWrGjv!LE4db?HEWb5mu*t@{kv|XwK8?npOshmzf=vZA@ zVSN9sL~!sn?r(AK)Q7Jk2(|M67Uy3I{eRy z_l&Y@A>;vjkWN5I2xvFFTLX0i+`{qz7C_@bo`ZUzDugfq4+>a3?1v%)O+YTd6@Ul7 zAfLfm=nhZ`)P~&v90$&UcF+yXm9sq!qCx3^9gzIcO|Y(js^Fj)Rvq>nQAHI92ap=P z10A4@prk+AGWCb`2)dQYFuR$|H6iDE8p}9a?#nV2}LBCoCf(Xi2@szia7#gY>b|l!-U`c}@ zLdhvQjc!BdLJvYvzzzngnw51yRYCqh4}$oRCy-z|v3Hc*d|?^Wj=l~18*E~*cR_kU z{XsxM1i{V*4GujHQ3DBpl2w4FgFR48Nma@HPgnyKoIEY-MqmMeY=I<%oG~l!f<+FN z1ZY^;10j4M4#HYXP zw5eJpA_y(>uLQ~OucgxDLuf}fVs272FaMxhn4xnDGIyLXnw>Xsd^J8XhcWIwIoQ9} z%FoSJTAGW(SRGwJwb=@pY7r$uQRK3Zd~XbxU)ts!4XsJrCycrWSI?e!IqwqIR8+Jh zlRjZ`UO1I!BtJR_2~7AbkbSm%XQqxEPkz6BTGWx8e}nQ=w7bZ|eVP4?*Tb!$(R)iC z9)&%bS*u(lXqzitAN)Oo=&Ytn>%Hzjc<5liuPi>zC_nw;Z0AE3Y$Jao_Q90R-gl~5 z_xAb2J%eArrC1CN4G$}-zVvCqF1;H;abAu6G*+PDHSYFx@Tdbfox*uEd3}BUyYY-l zTfEsOqsi#f9^FoLO;ChK<554qkri&Av~SIM*{fEYRE?vH7pTAOmu2pz3X?Wn*!ROX ztd54huAk&mFBemMooL33RV-*1f0Q3_(7hl$<#*|WF9P!;r;4_+X~k~uKEqdzZ$5Al zV63XN@)j$FN#cCD;ek1R#l zv%pGrhB~KWgoCj%GT?%{@@o(AJGt*PG#l3i>lhmb_twKH^EYvacVY-6bsCl5*^~L0 zonm@lk2UvvTKr2RS%}T>^~EYqdL1q4nD%0n&Xqr^cK^`J5W;lRRB^R-O8b&HENO||mo0xaD+S=I8RTlIfVgqN@SXDr2&-)we--K7w= zJVU8?Z+7k9dy;s;^gDkQa`0nz6N{T?(A&Iz)2!DEecLyRa&FI!id#5Z7B*O2=PsR0 zEvc|8{NS^)!d)MDX(97Xw}m&kEO@5jqRaDZ!+%`wYOI<23q|&js`&o4xvjP7D_xv@ z5hEwpsp{HezI9!~6O{~)lLR@oF7?J7i>1|5a~UuoN=q&6N}EJPV_GD`&M*v8Y`^2j zKII*d_@Fi$+i*YEW+Hbzn{iQk~yP z>7N{S4)r*!NwQ`(qcN#8SRQsNK6>{)X12nbF`*7#ecO7I)Q$uZsV+xS4E7aUn+U(K baj7?x%VD!5Cxk2YbYLNVeiXvvpMCWYo=by@ literal 0 HcmV?d00001 diff --git a/front-end/public/images/logo.svg b/front-end/public/images/logo.svg new file mode 100644 index 0000000..030170d --- /dev/null +++ b/front-end/public/images/logo.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/front-end/public/index.html b/front-end/public/index.html new file mode 100644 index 0000000..aa069f2 --- /dev/null +++ b/front-end/public/index.html @@ -0,0 +1,43 @@ + + + + + + + + + + + + + React App + + + +
+ + + diff --git a/front-end/public/logo192.png b/front-end/public/logo192.png new file mode 100644 index 0000000000000000000000000000000000000000..fc44b0a3796c0e0a64c3d858ca038bd4570465d9 GIT binary patch literal 5347 zcmZWtbyO6NvR-oO24RV%BvuJ&=?+<7=`LvyB&A_#M7mSDYw1v6DJkiYl9XjT!%$dLEBTQ8R9|wd3008in6lFF3GV-6mLi?MoP_y~}QUnaDCHI#t z7w^m$@6DI)|C8_jrT?q=f8D?0AM?L)Z}xAo^e^W>t$*Y0KlT5=@bBjT9kxb%-KNdk zeOS1tKO#ChhG7%{ApNBzE2ZVNcxbrin#E1TiAw#BlUhXllzhN$qWez5l;h+t^q#Eav8PhR2|T}y5kkflaK`ba-eoE+Z2q@o6P$)=&` z+(8}+-McnNO>e#$Rr{32ngsZIAX>GH??tqgwUuUz6kjns|LjsB37zUEWd|(&O!)DY zQLrq%Y>)Y8G`yYbYCx&aVHi@-vZ3|ebG!f$sTQqMgi0hWRJ^Wc+Ibv!udh_r%2|U) zPi|E^PK?UE!>_4`f`1k4hqqj_$+d!EB_#IYt;f9)fBOumGNyglU(ofY`yHq4Y?B%- zp&G!MRY<~ajTgIHErMe(Z8JG*;D-PJhd@RX@QatggM7+G(Lz8eZ;73)72Hfx5KDOE zkT(m}i2;@X2AT5fW?qVp?@WgN$aT+f_6eo?IsLh;jscNRp|8H}Z9p_UBO^SJXpZew zEK8fz|0Th%(Wr|KZBGTM4yxkA5CFdAj8=QSrT$fKW#tweUFqr0TZ9D~a5lF{)%-tTGMK^2tz(y2v$i%V8XAxIywrZCp=)83p(zIk6@S5AWl|Oa2hF`~~^W zI;KeOSkw1O#TiQ8;U7OPXjZM|KrnN}9arP)m0v$c|L)lF`j_rpG(zW1Qjv$=^|p*f z>)Na{D&>n`jOWMwB^TM}slgTEcjxTlUby89j1)|6ydRfWERn3|7Zd2&e7?!K&5G$x z`5U3uFtn4~SZq|LjFVrz$3iln-+ucY4q$BC{CSm7Xe5c1J<=%Oagztj{ifpaZk_bQ z9Sb-LaQMKp-qJA*bP6DzgE3`}*i1o3GKmo2pn@dj0;He}F=BgINo};6gQF8!n0ULZ zL>kC0nPSFzlcB7p41doao2F7%6IUTi_+!L`MM4o*#Y#0v~WiO8uSeAUNp=vA2KaR&=jNR2iVwG>7t%sG2x_~yXzY)7K& zk3p+O0AFZ1eu^T3s};B%6TpJ6h-Y%B^*zT&SN7C=N;g|#dGIVMSOru3iv^SvO>h4M=t-N1GSLLDqVTcgurco6)3&XpU!FP6Hlrmj}f$ zp95;b)>M~`kxuZF3r~a!rMf4|&1=uMG$;h^g=Kl;H&Np-(pFT9FF@++MMEx3RBsK?AU0fPk-#mdR)Wdkj)`>ZMl#^<80kM87VvsI3r_c@_vX=fdQ`_9-d(xiI z4K;1y1TiPj_RPh*SpDI7U~^QQ?%0&!$Sh#?x_@;ag)P}ZkAik{_WPB4rHyW#%>|Gs zdbhyt=qQPA7`?h2_8T;-E6HI#im9K>au*(j4;kzwMSLgo6u*}-K`$_Gzgu&XE)udQ zmQ72^eZd|vzI)~!20JV-v-T|<4@7ruqrj|o4=JJPlybwMg;M$Ud7>h6g()CT@wXm` zbq=A(t;RJ^{Xxi*Ff~!|3!-l_PS{AyNAU~t{h;(N(PXMEf^R(B+ZVX3 z8y0;0A8hJYp@g+c*`>eTA|3Tgv9U8#BDTO9@a@gVMDxr(fVaEqL1tl?md{v^j8aUv zm&%PX4^|rX|?E4^CkplWWNv*OKM>DxPa z!RJ)U^0-WJMi)Ksc!^ixOtw^egoAZZ2Cg;X7(5xZG7yL_;UJ#yp*ZD-;I^Z9qkP`} zwCTs0*%rIVF1sgLervtnUo&brwz?6?PXRuOCS*JI-WL6GKy7-~yi0giTEMmDs_-UX zo=+nFrW_EfTg>oY72_4Z0*uG>MnXP=c0VpT&*|rvv1iStW;*^={rP1y?Hv+6R6bxFMkxpWkJ>m7Ba{>zc_q zEefC3jsXdyS5??Mz7IET$Kft|EMNJIv7Ny8ZOcKnzf`K5Cd)&`-fTY#W&jnV0l2vt z?Gqhic}l}mCv1yUEy$%DP}4AN;36$=7aNI^*AzV(eYGeJ(Px-j<^gSDp5dBAv2#?; zcMXv#aj>%;MiG^q^$0MSg-(uTl!xm49dH!{X0){Ew7ThWV~Gtj7h%ZD zVN-R-^7Cf0VH!8O)uUHPL2mO2tmE*cecwQv_5CzWeh)ykX8r5Hi`ehYo)d{Jnh&3p z9ndXT$OW51#H5cFKa76c<%nNkP~FU93b5h-|Cb}ScHs@4Q#|}byWg;KDMJ#|l zE=MKD*F@HDBcX@~QJH%56eh~jfPO-uKm}~t7VkHxHT;)4sd+?Wc4* z>CyR*{w@4(gnYRdFq=^(#-ytb^5ESD?x<0Skhb%Pt?npNW1m+Nv`tr9+qN<3H1f<% zZvNEqyK5FgPsQ`QIu9P0x_}wJR~^CotL|n zk?dn;tLRw9jJTur4uWoX6iMm914f0AJfB@C74a;_qRrAP4E7l890P&{v<}>_&GLrW z)klculcg`?zJO~4;BBAa=POU%aN|pmZJn2{hA!d!*lwO%YSIzv8bTJ}=nhC^n}g(ld^rn#kq9Z3)z`k9lvV>y#!F4e{5c$tnr9M{V)0m(Z< z#88vX6-AW7T2UUwW`g<;8I$Jb!R%z@rCcGT)-2k7&x9kZZT66}Ztid~6t0jKb&9mm zpa}LCb`bz`{MzpZR#E*QuBiZXI#<`5qxx=&LMr-UUf~@dRk}YI2hbMsAMWOmDzYtm zjof16D=mc`^B$+_bCG$$@R0t;e?~UkF?7<(vkb70*EQB1rfUWXh$j)R2)+dNAH5%R zEBs^?N;UMdy}V};59Gu#0$q53$}|+q7CIGg_w_WlvE}AdqoS<7DY1LWS9?TrfmcvT zaypmplwn=P4;a8-%l^e?f`OpGb}%(_mFsL&GywhyN(-VROj`4~V~9bGv%UhcA|YW% zs{;nh@aDX11y^HOFXB$a7#Sr3cEtNd4eLm@Y#fc&j)TGvbbMwze zXtekX_wJqxe4NhuW$r}cNy|L{V=t#$%SuWEW)YZTH|!iT79k#?632OFse{+BT_gau zJwQcbH{b}dzKO?^dV&3nTILYlGw{27UJ72ZN){BILd_HV_s$WfI2DC<9LIHFmtyw? zQ;?MuK7g%Ym+4e^W#5}WDLpko%jPOC=aN)3!=8)s#Rnercak&b3ESRX3z{xfKBF8L z5%CGkFmGO@x?_mPGlpEej!3!AMddChabyf~nJNZxx!D&{@xEb!TDyvqSj%Y5@A{}9 zRzoBn0?x}=krh{ok3Nn%e)#~uh;6jpezhA)ySb^b#E>73e*frBFu6IZ^D7Ii&rsiU z%jzygxT-n*joJpY4o&8UXr2s%j^Q{?e-voloX`4DQyEK+DmrZh8A$)iWL#NO9+Y@!sO2f@rI!@jN@>HOA< z?q2l{^%mY*PNx2FoX+A7X3N}(RV$B`g&N=e0uvAvEN1W^{*W?zT1i#fxuw10%~))J zjx#gxoVlXREWZf4hRkgdHx5V_S*;p-y%JtGgQ4}lnA~MBz-AFdxUxU1RIT$`sal|X zPB6sEVRjGbXIP0U+?rT|y5+ev&OMX*5C$n2SBPZr`jqzrmpVrNciR0e*Wm?fK6DY& zl(XQZ60yWXV-|Ps!A{EF;=_z(YAF=T(-MkJXUoX zI{UMQDAV2}Ya?EisdEW;@pE6dt;j0fg5oT2dxCi{wqWJ<)|SR6fxX~5CzblPGr8cb zUBVJ2CQd~3L?7yfTpLNbt)He1D>*KXI^GK%<`bq^cUq$Q@uJifG>p3LU(!H=C)aEL zenk7pVg}0{dKU}&l)Y2Y2eFMdS(JS0}oZUuVaf2+K*YFNGHB`^YGcIpnBlMhO7d4@vV zv(@N}(k#REdul8~fP+^F@ky*wt@~&|(&&meNO>rKDEnB{ykAZ}k>e@lad7to>Ao$B zz<1(L=#J*u4_LB=8w+*{KFK^u00NAmeNN7pr+Pf+N*Zl^dO{LM-hMHyP6N!~`24jd zXYP|Ze;dRXKdF2iJG$U{k=S86l@pytLx}$JFFs8e)*Vi?aVBtGJ3JZUj!~c{(rw5>vuRF$`^p!P8w1B=O!skwkO5yd4_XuG^QVF z`-r5K7(IPSiKQ2|U9+`@Js!g6sfJwAHVd|s?|mnC*q zp|B|z)(8+mxXyxQ{8Pg3F4|tdpgZZSoU4P&9I8)nHo1@)9_9u&NcT^FI)6|hsAZFk zZ+arl&@*>RXBf-OZxhZerOr&dN5LW9@gV=oGFbK*J+m#R-|e6(Loz(;g@T^*oO)0R zN`N=X46b{7yk5FZGr#5&n1!-@j@g02g|X>MOpF3#IjZ_4wg{dX+G9eqS+Es9@6nC7 zD9$NuVJI}6ZlwtUm5cCAiYv0(Yi{%eH+}t)!E^>^KxB5^L~a`4%1~5q6h>d;paC9c zTj0wTCKrhWf+F#5>EgX`sl%POl?oyCq0(w0xoL?L%)|Q7d|Hl92rUYAU#lc**I&^6p=4lNQPa0 znQ|A~i0ip@`B=FW-Q;zh?-wF;Wl5!+q3GXDu-x&}$gUO)NoO7^$BeEIrd~1Dh{Tr` z8s<(Bn@gZ(mkIGnmYh_ehXnq78QL$pNDi)|QcT*|GtS%nz1uKE+E{7jdEBp%h0}%r zD2|KmYGiPa4;md-t_m5YDz#c*oV_FqXd85d@eub?9N61QuYcb3CnVWpM(D-^|CmkL z(F}L&N7qhL2PCq)fRh}XO@U`Yn<?TNGR4L(mF7#4u29{i~@k;pLsgl({YW5`Mo+p=zZn3L*4{JU;++dG9 X@eDJUQo;Ye2mwlRs?y0|+_a0zY+Zo%Dkae}+MySoIppb75o?vUW_?)>@g{U2`ERQIXV zeY$JrWnMZ$QC<=ii4X|@0H8`si75jB(ElJb00HAB%>SlLR{!zO|C9P3zxw_U8?1d8uRZ=({Ga4shyN}3 zAK}WA(ds|``G4jA)9}Bt2Hy0+f3rV1E6b|@?hpGA=PI&r8)ah|)I2s(P5Ic*Ndhn^ z*T&j@gbCTv7+8rpYbR^Ty}1AY)YH;p!m948r#%7x^Z@_-w{pDl|1S4`EM3n_PaXvK z1JF)E3qy$qTj5Xs{jU9k=y%SQ0>8E$;x?p9ayU0bZZeo{5Z@&FKX>}s!0+^>C^D#z z>xsCPvxD3Z=dP}TTOSJhNTPyVt14VCQ9MQFN`rn!c&_p?&4<5_PGm4a;WS&1(!qKE z_H$;dDdiPQ!F_gsN`2>`X}$I=B;={R8%L~`>RyKcS$72ai$!2>d(YkciA^J0@X%G4 z4cu!%Ps~2JuJ8ex`&;Fa0NQOq_nDZ&X;^A=oc1&f#3P1(!5il>6?uK4QpEG8z0Rhu zvBJ+A9RV?z%v?!$=(vcH?*;vRs*+PPbOQ3cdPr5=tOcLqmfx@#hOqX0iN)wTTO21jH<>jpmwRIAGw7`a|sl?9y9zRBh>(_%| zF?h|P7}~RKj?HR+q|4U`CjRmV-$mLW>MScKnNXiv{vD3&2@*u)-6P@h0A`eeZ7}71 zK(w%@R<4lLt`O7fs1E)$5iGb~fPfJ?WxhY7c3Q>T-w#wT&zW522pH-B%r5v#5y^CF zcC30Se|`D2mY$hAlIULL%-PNXgbbpRHgn<&X3N9W!@BUk@9g*P5mz-YnZBb*-$zMM z7Qq}ic0mR8n{^L|=+diODdV}Q!gwr?y+2m=3HWwMq4z)DqYVg0J~^}-%7rMR@S1;9 z7GFj6K}i32X;3*$SmzB&HW{PJ55kT+EI#SsZf}bD7nW^Haf}_gXciYKX{QBxIPSx2Ma? zHQqgzZq!_{&zg{yxqv3xq8YV+`S}F6A>Gtl39_m;K4dA{pP$BW0oIXJ>jEQ!2V3A2 zdpoTxG&V=(?^q?ZTj2ZUpDUdMb)T?E$}CI>r@}PFPWD9@*%V6;4Ag>D#h>!s)=$0R zRXvdkZ%|c}ubej`jl?cS$onl9Tw52rBKT)kgyw~Xy%z62Lr%V6Y=f?2)J|bZJ5(Wx zmji`O;_B+*X@qe-#~`HFP<{8$w@z4@&`q^Q-Zk8JG3>WalhnW1cvnoVw>*R@c&|o8 zZ%w!{Z+MHeZ*OE4v*otkZqz11*s!#s^Gq>+o`8Z5 z^i-qzJLJh9!W-;SmFkR8HEZJWiXk$40i6)7 zZpr=k2lp}SasbM*Nbn3j$sn0;rUI;%EDbi7T1ZI4qL6PNNM2Y%6{LMIKW+FY_yF3) zSKQ2QSujzNMSL2r&bYs`|i2Dnn z=>}c0>a}>|uT!IiMOA~pVT~R@bGlm}Edf}Kq0?*Af6#mW9f9!}RjW7om0c9Qlp;yK z)=XQs(|6GCadQbWIhYF=rf{Y)sj%^Id-ARO0=O^Ad;Ph+ z0?$eE1xhH?{T$QI>0JP75`r)U_$#%K1^BQ8z#uciKf(C701&RyLQWBUp*Q7eyn76} z6JHpC9}R$J#(R0cDCkXoFSp;j6{x{b&0yE@P7{;pCEpKjS(+1RQy38`=&Yxo%F=3y zCPeefABp34U-s?WmU#JJw23dcC{sPPFc2#J$ZgEN%zod}J~8dLm*fx9f6SpO zn^Ww3bt9-r0XaT2a@Wpw;C23XM}7_14#%QpubrIw5aZtP+CqIFmsG4`Cm6rfxl9n5 z7=r2C-+lM2AB9X0T_`?EW&Byv&K?HS4QLoylJ|OAF z`8atBNTzJ&AQ!>sOo$?^0xj~D(;kS$`9zbEGd>f6r`NC3X`tX)sWgWUUOQ7w=$TO&*j;=u%25ay-%>3@81tGe^_z*C7pb9y*Ed^H3t$BIKH2o+olp#$q;)_ zfpjCb_^VFg5fU~K)nf*d*r@BCC>UZ!0&b?AGk_jTPXaSnCuW110wjHPPe^9R^;jo3 zwvzTl)C`Zl5}O2}3lec=hZ*$JnkW#7enKKc)(pM${_$9Hc=Sr_A9Biwe*Y=T?~1CK z6eZ9uPICjy-sMGbZl$yQmpB&`ouS8v{58__t0$JP%i3R&%QR3ianbZqDs<2#5FdN@n5bCn^ZtH992~5k(eA|8|@G9u`wdn7bnpg|@{m z^d6Y`*$Zf2Xr&|g%sai#5}Syvv(>Jnx&EM7-|Jr7!M~zdAyjt*xl;OLhvW-a%H1m0 z*x5*nb=R5u><7lyVpNAR?q@1U59 zO+)QWwL8t zyip?u_nI+K$uh{y)~}qj?(w0&=SE^8`_WMM zTybjG=999h38Yes7}-4*LJ7H)UE8{mE(6;8voE+TYY%33A>S6`G_95^5QHNTo_;Ao ztIQIZ_}49%{8|=O;isBZ?=7kfdF8_@azfoTd+hEJKWE!)$)N%HIe2cplaK`ry#=pV z0q{9w-`i0h@!R8K3GC{ivt{70IWG`EP|(1g7i_Q<>aEAT{5(yD z=!O?kq61VegV+st@XCw475j6vS)_z@efuqQgHQR1T4;|-#OLZNQJPV4k$AX1Uk8Lm z{N*b*ia=I+MB}kWpupJ~>!C@xEN#Wa7V+7{m4j8c?)ChV=D?o~sjT?0C_AQ7B-vxqX30s0I_`2$in86#`mAsT-w?j{&AL@B3$;P z31G4(lV|b}uSDCIrjk+M1R!X7s4Aabn<)zpgT}#gE|mIvV38^ODy@<&yflpCwS#fRf9ZX3lPV_?8@C5)A;T zqmouFLFk;qIs4rA=hh=GL~sCFsXHsqO6_y~*AFt939UYVBSx1s(=Kb&5;j7cSowdE;7()CC2|-i9Zz+_BIw8#ll~-tyH?F3{%`QCsYa*b#s*9iCc`1P1oC26?`g<9))EJ3%xz+O!B3 zZ7$j~To)C@PquR>a1+Dh>-a%IvH_Y7^ys|4o?E%3`I&ADXfC8++hAdZfzIT#%C+Jz z1lU~K_vAm0m8Qk}K$F>|>RPK%<1SI0(G+8q~H zAsjezyP+u!Se4q3GW)`h`NPSRlMoBjCzNPesWJwVTY!o@G8=(6I%4XHGaSiS3MEBK zhgGFv6Jc>L$4jVE!I?TQuwvz_%CyO!bLh94nqK11C2W$*aa2ueGopG8DnBICVUORP zgytv#)49fVXDaR$SukloYC3u7#5H)}1K21=?DKj^U)8G;MS)&Op)g^zR2($<>C*zW z;X7`hLxiIO#J`ANdyAOJle4V%ppa*(+0i3w;8i*BA_;u8gOO6)MY`ueq7stBMJTB; z-a0R>hT*}>z|Gg}@^zDL1MrH+2hsR8 zHc}*9IvuQC^Ju)^#Y{fOr(96rQNPNhxc;mH@W*m206>Lo<*SaaH?~8zg&f&%YiOEG zGiz?*CP>Bci}!WiS=zj#K5I}>DtpregpP_tfZtPa(N<%vo^#WCQ5BTv0vr%Z{)0q+ z)RbfHktUm|lg&U3YM%lMUM(fu}i#kjX9h>GYctkx9Mt_8{@s%!K_EI zScgwy6%_fR?CGJQtmgNAj^h9B#zmaMDWgH55pGuY1Gv7D z;8Psm(vEPiwn#MgJYu4Ty9D|h!?Rj0ddE|&L3S{IP%H4^N!m`60ZwZw^;eg4sk6K{ ziA^`Sbl_4~f&Oo%n;8Ye(tiAdlZKI!Z=|j$5hS|D$bDJ}p{gh$KN&JZYLUjv4h{NY zBJ>X9z!xfDGY z+oh_Z&_e#Q(-}>ssZfm=j$D&4W4FNy&-kAO1~#3Im;F)Nwe{(*75(p=P^VI?X0GFakfh+X-px4a%Uw@fSbmp9hM1_~R>?Z8+ ziy|e9>8V*`OP}4x5JjdWp}7eX;lVxp5qS}0YZek;SNmm7tEeSF*-dI)6U-A%m6YvCgM(}_=k#a6o^%-K4{`B1+}O4x zztDT%hVb;v#?j`lTvlFQ3aV#zkX=7;YFLS$uIzb0E3lozs5`Xy zi~vF+%{z9uLjKvKPhP%x5f~7-Gj+%5N`%^=yk*Qn{`> z;xj&ROY6g`iy2a@{O)V(jk&8#hHACVDXey5a+KDod_Z&}kHM}xt7}Md@pil{2x7E~ zL$k^d2@Ec2XskjrN+IILw;#7((abu;OJii&v3?60x>d_Ma(onIPtcVnX@ELF0aL?T zSmWiL3(dOFkt!x=1O!_0n(cAzZW+3nHJ{2S>tgSK?~cFha^y(l@-Mr2W$%MN{#af8J;V*>hdq!gx=d0h$T7l}>91Wh07)9CTX zh2_ZdQCyFOQ)l(}gft0UZG`Sh2`x-w`5vC2UD}lZs*5 zG76$akzn}Xi))L3oGJ75#pcN=cX3!=57$Ha=hQ2^lwdyU#a}4JJOz6ddR%zae%#4& za)bFj)z=YQela(F#Y|Q#dp}PJghITwXouVaMq$BM?K%cXn9^Y@g43$=O)F&ZlOUom zJiad#dea;-eywBA@e&D6Pdso1?2^(pXiN91?jvcaUyYoKUmvl5G9e$W!okWe*@a<^ z8cQQ6cNSf+UPDx%?_G4aIiybZHHagF{;IcD(dPO!#=u zWfqLcPc^+7Uu#l(Bpxft{*4lv#*u7X9AOzDO z1D9?^jIo}?%iz(_dwLa{ex#T}76ZfN_Z-hwpus9y+4xaUu9cX}&P{XrZVWE{1^0yw zO;YhLEW!pJcbCt3L8~a7>jsaN{V3>tz6_7`&pi%GxZ=V3?3K^U+*ryLSb)8^IblJ0 zSRLNDvIxt)S}g30?s_3NX>F?NKIGrG_zB9@Z>uSW3k2es_H2kU;Rnn%j5qP)!XHKE zPB2mHP~tLCg4K_vH$xv`HbRsJwbZMUV(t=ez;Ec(vyHH)FbfLg`c61I$W_uBB>i^r z&{_P;369-&>23R%qNIULe=1~T$(DA`ev*EWZ6j(B$(te}x1WvmIll21zvygkS%vwG zzkR6Z#RKA2!z!C%M!O>!=Gr0(J0FP=-MN=5t-Ir)of50y10W}j`GtRCsXBakrKtG& zazmITDJMA0C51&BnLY)SY9r)NVTMs);1<=oosS9g31l{4ztjD3#+2H7u_|66b|_*O z;Qk6nalpqdHOjx|K&vUS_6ITgGll;TdaN*ta=M_YtyC)I9Tmr~VaPrH2qb6sd~=AcIxV+%z{E&0@y=DPArw zdV7z(G1hBx7hd{>(cr43^WF%4Y@PXZ?wPpj{OQ#tvc$pABJbvPGvdR`cAtHn)cSEV zrpu}1tJwQ3y!mSmH*uz*x0o|CS<^w%&KJzsj~DU0cLQUxk5B!hWE>aBkjJle8z~;s z-!A=($+}Jq_BTK5^B!`R>!MulZN)F=iXXeUd0w5lUsE5VP*H*oCy(;?S$p*TVvTxwAeWFB$jHyb0593)$zqalVlDX=GcCN1gU0 zlgU)I$LcXZ8Oyc2TZYTPu@-;7<4YYB-``Qa;IDcvydIA$%kHhJKV^m*-zxcvU4viy&Kr5GVM{IT>WRywKQ9;>SEiQD*NqplK-KK4YR`p0@JW)n_{TU3bt0 zim%;(m1=#v2}zTps=?fU5w^(*y)xT%1vtQH&}50ZF!9YxW=&7*W($2kgKyz1mUgfs zfV<*XVVIFnohW=|j+@Kfo!#liQR^x>2yQdrG;2o8WZR+XzU_nG=Ed2rK?ntA;K5B{ z>M8+*A4!Jm^Bg}aW?R?6;@QG@uQ8&oJ{hFixcfEnJ4QH?A4>P=q29oDGW;L;= z9-a0;g%c`C+Ai!UmK$NC*4#;Jp<1=TioL=t^YM)<<%u#hnnfSS`nq63QKGO1L8RzX z@MFDqs1z ztYmxDl@LU)5acvHk)~Z`RW7=aJ_nGD!mOSYD>5Odjn@TK#LY{jf?+piB5AM-CAoT_ z?S-*q7}wyLJzK>N%eMPuFgN)Q_otKP;aqy=D5f!7<=n(lNkYRXVpkB{TAYLYg{|(jtRqYmg$xH zjmq?B(RE4 zQx^~Pt}gxC2~l=K$$-sYy_r$CO(d=+b3H1MB*y_5g6WLaWTXn+TKQ|hNY^>Mp6k*$ zwkovomhu776vQATqT4blf~g;TY(MWCrf^^yfWJvSAB$p5l;jm@o#=!lqw+Lqfq>X= z$6~kxfm7`3q4zUEB;u4qa#BdJxO!;xGm)wwuisj{0y2x{R(IGMrsIzDY9LW>m!Y`= z04sx3IjnYvL<4JqxQ8f7qYd0s2Ig%`ytYPEMKI)s(LD}D@EY>x`VFtqvnADNBdeao zC96X+MxnwKmjpg{U&gP3HE}1=s!lv&D{6(g_lzyF3A`7Jn*&d_kL<;dAFx!UZ>hB8 z5A*%LsAn;VLp>3${0>M?PSQ)9s3}|h2e?TG4_F{}{Cs>#3Q*t$(CUc}M)I}8cPF6% z=+h(Kh^8)}gj(0}#e7O^FQ6`~fd1#8#!}LMuo3A0bN`o}PYsm!Y}sdOz$+Tegc=qT z8x`PH$7lvnhJp{kHWb22l;@7B7|4yL4UOOVM0MP_>P%S1Lnid)+k9{+3D+JFa#Pyf zhVc#&df87APl4W9X)F3pGS>@etfl=_E5tBcVoOfrD4hmVeTY-cj((pkn%n@EgN{0f zwb_^Rk0I#iZuHK!l*lN`ceJn(sI{$Fq6nN& zE<-=0_2WN}m+*ivmIOxB@#~Q-cZ>l136w{#TIJe478`KE7@=a{>SzPHsKLzYAyBQO zAtuuF$-JSDy_S@6GW0MOE~R)b;+0f%_NMrW(+V#c_d&U8Z9+ec4=HmOHw?gdjF(Lu zzra83M_BoO-1b3;9`%&DHfuUY)6YDV21P$C!Rc?mv&{lx#f8oc6?0?x zK08{WP65?#>(vPfA-c=MCY|%*1_<3D4NX zeVTi-JGl2uP_2@0F{G({pxQOXt_d{g_CV6b?jNpfUG9;8yle-^4KHRvZs-_2siata zt+d_T@U$&t*xaD22(fH(W1r$Mo?3dc%Tncm=C6{V9y{v&VT#^1L04vDrLM9qBoZ4@ z6DBN#m57hX7$C(=#$Y5$bJmwA$T8jKD8+6A!-IJwA{WOfs%s}yxUw^?MRZjF$n_KN z6`_bGXcmE#5e4Ym)aQJ)xg3Pg0@k`iGuHe?f(5LtuzSq=nS^5z>vqU0EuZ&75V%Z{ zYyhRLN^)$c6Ds{f7*FBpE;n5iglx5PkHfWrj3`x^j^t z7ntuV`g!9Xg#^3!x)l*}IW=(Tz3>Y5l4uGaB&lz{GDjm2D5S$CExLT`I1#n^lBH7Y zDgpMag@`iETKAI=p<5E#LTkwzVR@=yY|uBVI1HG|8h+d;G-qfuj}-ZR6fN>EfCCW z9~wRQoAPEa#aO?3h?x{YvV*d+NtPkf&4V0k4|L=uj!U{L+oLa(z#&iuhJr3-PjO3R z5s?=nn_5^*^Rawr>>Nr@K(jwkB#JK-=+HqwfdO<+P5byeim)wvqGlP-P|~Nse8=XF zz`?RYB|D6SwS}C+YQv+;}k6$-%D(@+t14BL@vM z2q%q?f6D-A5s$_WY3{^G0F131bbh|g!}#BKw=HQ7mx;Dzg4Z*bTLQSfo{ed{4}NZW zfrRm^Ca$rlE{Ue~uYv>R9{3smwATcdM_6+yWIO z*ZRH~uXE@#p$XTbCt5j7j2=86e{9>HIB6xDzV+vAo&B?KUiMP|ttOElepnl%|DPqL b{|{}U^kRn2wo}j7|0ATu<;8xA7zX}7|B6mN literal 0 HcmV?d00001 diff --git a/front-end/public/manifest.json b/front-end/public/manifest.json new file mode 100644 index 0000000..080d6c7 --- /dev/null +++ b/front-end/public/manifest.json @@ -0,0 +1,25 @@ +{ + "short_name": "React App", + "name": "Create React App Sample", + "icons": [ + { + "src": "favicon.ico", + "sizes": "64x64 32x32 24x24 16x16", + "type": "image/x-icon" + }, + { + "src": "logo192.png", + "type": "image/png", + "sizes": "192x192" + }, + { + "src": "logo512.png", + "type": "image/png", + "sizes": "512x512" + } + ], + "start_url": ".", + "display": "standalone", + "theme_color": "#000000", + "background_color": "#ffffff" +} diff --git a/front-end/public/robots.txt b/front-end/public/robots.txt new file mode 100644 index 0000000..e9e57dc --- /dev/null +++ b/front-end/public/robots.txt @@ -0,0 +1,3 @@ +# https://www.robotstxt.org/robotstxt.html +User-agent: * +Disallow: diff --git a/front-end/src/App.css b/front-end/src/App.css new file mode 100644 index 0000000..0c24d01 --- /dev/null +++ b/front-end/src/App.css @@ -0,0 +1,5 @@ +@import "./styles/variables.css"; + +input { + font-family: 'Helvetica', sans-serif; +} \ No newline at end of file diff --git a/front-end/src/App.js b/front-end/src/App.js new file mode 100644 index 0000000..c199949 --- /dev/null +++ b/front-end/src/App.js @@ -0,0 +1,216 @@ +import React, { useState, useEffect, useRef } from "react"; +import { PaperPlaneIcon } from "@radix-ui/react-icons"; + +const BASE_URL = "http://localhost:8080"; +const USER_ID = "12345678-0000-0000-0000-000000000000"; + +export default function RAGChatUI() { + const [messages, setMessages] = useState([]); + const [input, setInput] = useState(""); + const [loading, setLoading] = useState(false); + const [token, setToken] = useState(""); + const [chatId, setChatId] = useState(""); + const messagesEndRef = useRef(null); + + useEffect(() => { + messagesEndRef.current?.scrollIntoView({ behavior: "smooth" }); + }, [messages]); + + useEffect(() => { + const authenticateAndStartChat = async () => { + try { + const tokenResponse = await fetch(`${BASE_URL}/token`, { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ username: "user", password: "password" }), + }); + + const tokenData = await tokenResponse.json(); + if (!tokenData.token) throw new Error("Token retrieval failed."); + setToken(tokenData.token); + + const chatResponse = await fetch(`${BASE_URL}/users/${USER_ID}/chat-sessions`, { + method: "POST", + headers: { Authorization: `Bearer ${tokenData.token}` }, + }); + + const chatData = await chatResponse.json(); + if (!chatData.id) throw new Error("Chat session creation failed."); + setChatId(chatData.id); + } catch (err) { + console.error(err); + } + }; + + authenticateAndStartChat(); + }, []); + + const sendMessage = async () => { + if (!input.trim() || !token || !chatId) return; + + const userMessage = { role: "user", content: input }; + setMessages([userMessage, { role: "assistant", content: "" }]); + const prompt = input; + setInput(""); + setLoading(true); + + try { + const response = await fetch( + `${BASE_URL}/sse/users/${USER_ID}/chat-sessions/${chatId}/messages`, + { + method: "POST", + headers: { + "Content-Type": "application/json", + Authorization: `Bearer ${token}`, + }, + body: JSON.stringify({ content: prompt }), + } + ); + + if (!response.ok || !response.body) { + throw new Error("Streaming failed"); + } + + const reader = response.body.getReader(); + const decoder = new TextDecoder("utf-8"); + let fullMessage = ""; + let buffer = ""; + + while (true) { + const { value, done } = await reader.read(); + if (done) break; + buffer += decoder.decode(value, { stream: true }); + + const lines = buffer.split(/\n/); + buffer = lines.pop() || ""; + + for (const line of lines) { + if (line.startsWith("data:")) { + const json = line.slice(5).trim(); + try { + const parsed = JSON.parse(json); + if (parsed.token) { + fullMessage += parsed.token + " "; + let extracted = ""; + const contentMatch = fullMessage.match(/Content:(.*?)CreatedAt:/); + if (contentMatch) { + extracted = contentMatch[1].trim(); + } else if (fullMessage.includes("Content:")) { + extracted = fullMessage.split("Content:").pop().trim(); + } else { + extracted = ""; + } + setMessages((prev) => [prev[0], { role: "assistant", content: extracted }]); + } + } catch (e) { + console.warn("Skipping invalid JSON:", json); + } + } + } + } + } catch (err) { + console.error("Streaming message failed:", err); + setMessages([messages[0], { role: "assistant", content: "Error retrieving response." }]); + } + + setLoading(false); + }; + + const handleKeyDown = (e) => { + if (e.key === "Enter" && !e.shiftKey) { + e.preventDefault(); + sendMessage(); + } + }; + + return ( +
+

+ Louk Chatwalker +

+
+ + +
+ {messages[1]?.content || "The answer will appear here..."} +
+
+
+ ); +} diff --git a/front-end/src/App.test.js b/front-end/src/App.test.js new file mode 100644 index 0000000..1f03afe --- /dev/null +++ b/front-end/src/App.test.js @@ -0,0 +1,8 @@ +import { render, screen } from '@testing-library/react'; +import App from './App'; + +test('renders learn react link', () => { + render(); + const linkElement = screen.getByText(/learn react/i); + expect(linkElement).toBeInTheDocument(); +}); diff --git a/front-end/src/index.css b/front-end/src/index.css new file mode 100644 index 0000000..ec2585e --- /dev/null +++ b/front-end/src/index.css @@ -0,0 +1,13 @@ +body { + margin: 0; + font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen', + 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', + sans-serif; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +code { + font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', + monospace; +} diff --git a/front-end/src/index.js b/front-end/src/index.js new file mode 100644 index 0000000..e964cf4 --- /dev/null +++ b/front-end/src/index.js @@ -0,0 +1,20 @@ +import React from 'react'; +import ReactDOM from 'react-dom/client'; +import './index.css'; +import App from './App'; +import reportWebVitals from './reportWebVitals'; +import {BrowserRouter} from "react-router-dom"; + +const root = ReactDOM.createRoot(document.getElementById('root')); +root.render( + + + + + +); + +// If you want to start measuring performance in your app, pass a function +// to log results (for example: reportWebVitals(console.log)) +// or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals +// reportWebVitals(); diff --git a/front-end/src/logo.svg b/front-end/src/logo.svg new file mode 100644 index 0000000..9dfc1c0 --- /dev/null +++ b/front-end/src/logo.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/front-end/src/reportWebVitals.js b/front-end/src/reportWebVitals.js new file mode 100644 index 0000000..5253d3a --- /dev/null +++ b/front-end/src/reportWebVitals.js @@ -0,0 +1,13 @@ +const reportWebVitals = onPerfEntry => { + if (onPerfEntry && onPerfEntry instanceof Function) { + import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => { + getCLS(onPerfEntry); + getFID(onPerfEntry); + getFCP(onPerfEntry); + getLCP(onPerfEntry); + getTTFB(onPerfEntry); + }); + } +}; + +export default reportWebVitals; diff --git a/front-end/src/routes/ProtectedRoute.js b/front-end/src/routes/ProtectedRoute.js new file mode 100644 index 0000000..b1e54d8 --- /dev/null +++ b/front-end/src/routes/ProtectedRoute.js @@ -0,0 +1,10 @@ +import React from "react"; +import {Navigate} from "react-router-dom"; +import Cookies from "universal-cookie"; + +const cookies = new Cookies(); + +export default function ProtectedRoutes({children}) { + const token = cookies.get("access_token"); + return token ? <>{children} : ; +} \ No newline at end of file diff --git a/front-end/src/setupTests.js b/front-end/src/setupTests.js new file mode 100644 index 0000000..8f2609b --- /dev/null +++ b/front-end/src/setupTests.js @@ -0,0 +1,5 @@ +// jest-dom adds custom jest matchers for asserting on DOM nodes. +// allows you to do things like: +// expect(element).toHaveTextContent(/react/i) +// learn more: https://github.com/testing-library/jest-dom +import '@testing-library/jest-dom'; diff --git a/front-end/src/styles/variables.css b/front-end/src/styles/variables.css new file mode 100644 index 0000000..c4d6d38 --- /dev/null +++ b/front-end/src/styles/variables.css @@ -0,0 +1,15 @@ +:root { + --brand-color: #6070AC; + --brand-dark-color: #624c94; + --dark-color: #220D6A; + --light-color: #f2e7fe; + --opposite-color: #5ec7cc; + --positive-color: #60cc5e; + --negative-color: #f9a26e; + --dark-grey-color: #C3C3C3; + --text-grey-color: #696969; + --grey-color: #E0E0E0; + --mid-white-color: #F9F9FC; + --light-white-color: #ffffff; + --error-color: #ff3333; +} \ No newline at end of file