diff --git a/.coveralls.yml b/.coveralls.yml
new file mode 100644
index 0000000..90cf2a6
--- /dev/null
+++ b/.coveralls.yml
@@ -0,0 +1,3 @@
+service_name: travis-ci
+coverage_clover: build/logs/clover.xml
+json_path: build/logs/coveralls-upload.json
\ No newline at end of file
diff --git a/.dockerignore b/.dockerignore
new file mode 100644
index 0000000..8d87b1d
--- /dev/null
+++ b/.dockerignore
@@ -0,0 +1 @@
+node_modules/*
diff --git a/.editorconfig b/.editorconfig
new file mode 100644
index 0000000..95d26b0
--- /dev/null
+++ b/.editorconfig
@@ -0,0 +1,18 @@
+root = true
+
+[*]
+end_of_line = lf
+insert_final_newline = true
+charset = utf-8
+
+[*.{yaml,yml,yaml.dist,yml.dist}]
+indent_style = space
+indent_size = 2
+
+[*.json]
+indent_style = space
+indent_size = 4
+
+[*.{ts,tsx}]
+indent_style = space
+indent_size = 2
diff --git a/.env b/.env
new file mode 100644
index 0000000..c04bf31
--- /dev/null
+++ b/.env
@@ -0,0 +1,33 @@
+# In all environments, the following files are loaded if they exist,
+# the latter taking precedence over the former:
+#
+# * .env contains default values for the environment variables needed by the app
+# * .env.local uncommitted file with local overrides
+# * .env.$APP_ENV committed environment-specific defaults
+# * .env.$APP_ENV.local uncommitted environment-specific overrides
+#
+# Real environment variables win over .env files.
+#
+# DO NOT DEFINE PRODUCTION SECRETS IN THIS FILE NOR IN ANY OTHER COMMITTED FILES.
+# https://symfony.com/doc/current/configuration/secrets.html
+#
+# Run "composer dump-env prod" to compile .env files for production use (requires symfony/flex >=1.2).
+# https://symfony.com/doc/current/best_practices.html#use-environment-variables-for-infrastructure-configuration
+
+DOMAIN_NAME=rashin.me.local
+FRONTEND_PUBLIC_PORT=3121
+FRONTEND_ADMIN_PORT=3122
+
+APP_ENV=dev
+APP_SECRET=754b0fc1f6b595e6bd42fb1cc2af40f9
+
+# database configuration
+DB_HOST=db
+DB_PORT=5432
+DB_USER=user
+DB_PASSWORD=pwd
+DB_NAME=dbname
+DB_DRIVER=pdo_pgsql
+DB_SERVER_VERSION=15.2
+# PUBLIC DATABASE PORT
+DB_PUBLIC_PORT=5433
diff --git a/.env.test b/.env.test
new file mode 100644
index 0000000..733b902
--- /dev/null
+++ b/.env.test
@@ -0,0 +1,6 @@
+# define your env variables for the test env here
+KERNEL_CLASS='RashinMe\Kernel'
+APP_SECRET='$ecretf0rt3st'
+SYMFONY_DEPRECATIONS_HELPER=999999
+PANTHER_APP_ENV=panther
+PANTHER_ERROR_SCREENSHOT_DIR=./var/error-screenshots
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..600777f
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,16 @@
+/.env.local
+/.env.local.php
+/.env.*.local
+/config/secrets/prod/prod.decrypt.private.php
+/public/bundles/
+/var/
+/vendor/
+
+/.phpcs-cache
+/phpcs.xml
+
+
+.idea
+/phpunit.xml
+.phpunit.result.cache
+.phpunit.cache
diff --git a/.travis.yml b/.travis.yml
new file mode 100644
index 0000000..8756a74
--- /dev/null
+++ b/.travis.yml
@@ -0,0 +1,15 @@
+language: php
+php:
+ - 8.2
+
+before_script:
+ - composer self-update
+ - composer install --no-interaction
+ - mkdir -p build/logs
+
+script:
+ - ./vendor/bin/phpunit --coverage-clover build/logs/clover.xml
+ - ./vendor/bin/phpcs
+
+after_success:
+ - php vendor/bin/coveralls -v
diff --git a/Dockerfile b/Dockerfile
new file mode 100644
index 0000000..456eb86
--- /dev/null
+++ b/Dockerfile
@@ -0,0 +1,64 @@
+FROM php:8.2-fpm AS base
+
+ARG APP_ENV=dev
+ENV APP_ENV=$APP_ENV
+
+RUN apt-get update && apt-get install -y --no-install-recommends \
+ apt-utils \
+ zip \
+ unzip \
+ ssh \
+ g++ \
+ git \
+ curl \
+ libcurl4-gnutls-dev \
+ libpq-dev \
+ libicu-dev
+
+RUN docker-php-ext-install \
+ intl \
+ curl \
+ bcmath \
+ gettext \
+ pdo_pgsql
+
+
+RUN pecl install apcu \
+ && docker-php-ext-enable apcu
+
+RUN docker-php-ext-install opcache
+
+RUN pecl install -o -f redis \
+ && rm -rf /tmp/pear
+
+RUN docker-php-ext-enable redis
+
+RUN curl -sS https://getcomposer.org/installer | php \
+ && mv composer.phar /usr/local/bin/ \
+ && ln -s /usr/local/bin/composer.phar /usr/local/bin/composer
+
+
+FROM base as dev
+
+RUN pecl install xdebug \
+ && docker-php-ext-enable xdebug \
+ && echo "xdebug.mode=debug" >> /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini \
+ && echo "xdebug.start_with_request = yes" >> /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini \
+ && echo "xdebug.client_host=docker.for.mac.localhost" >> /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini \
+ && echo "xdebug.client_port=9001" >> /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini \
+ && echo "xdebug.log=/var/log/xdebug.log" >> /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini \
+ && echo "xdebug.idekey = PHPSTORM" >> /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini \
+ && touch /var/log/xdebug.log \
+ && chown www-data:www-data /var/log/xdebug.log \
+ && chmod 666 /var/log/xdebug.log
+
+WORKDIR /app
+
+CMD php-fpm -F
+
+FROM base as prod
+
+COPY ./ /app
+
+RUN composer install --no-dev --prefer-dist --no-progress --optimize-autoloader
+RUN php bin/console cache:clear
diff --git a/LICENSE.md b/LICENSE.md
new file mode 100644
index 0000000..d957729
--- /dev/null
+++ b/LICENSE.md
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2022 Mayank Agarwal
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
diff --git a/bin/console b/bin/console
new file mode 100755
index 0000000..77e88a3
--- /dev/null
+++ b/bin/console
@@ -0,0 +1,17 @@
+#!/usr/bin/env php
+run($GLOBALS['argv']);
+} else {
+ if (!is_file(dirname(__DIR__).'/vendor/symfony/phpunit-bridge/bin/simple-phpunit.php')) {
+ echo "Unable to find the `simple-phpunit.php` script in `vendor/symfony/phpunit-bridge/bin/`.\n";
+ exit(1);
+ }
+
+ require dirname(__DIR__).'/vendor/symfony/phpunit-bridge/bin/simple-phpunit.php';
+}
diff --git a/composer.json b/composer.json
new file mode 100644
index 0000000..a9c7dc9
--- /dev/null
+++ b/composer.json
@@ -0,0 +1,82 @@
+{
+ "type": "project",
+ "license": "proprietary",
+ "minimum-stability": "dev",
+ "prefer-stable": true,
+ "require": {
+ "php": "^8.2",
+ "ext-ctype": "*",
+ "ext-iconv": "*",
+ "doctrine/doctrine-bundle": "^2.8",
+ "doctrine/doctrine-migrations-bundle": "^3.2",
+ "doctrine/orm": "^2.14",
+ "dompdf/dompdf": "^2.0",
+ "ser/dto-request-bundle": "^0.1.0",
+ "symfony/console": "6.2.*",
+ "symfony/dotenv": "6.2.*",
+ "symfony/flex": "^2",
+ "symfony/framework-bundle": "6.2.*",
+ "symfony/mime": "6.2.*",
+ "symfony/runtime": "6.2.*",
+ "symfony/security-bundle": "6.2.*",
+ "symfony/validator": "6.2.*",
+ "symfony/yaml": "6.2.*"
+ },
+ "require-dev": {
+ "phpstan/phpstan": "^1.10",
+ "phpunit/phpunit": "^10.0",
+ "squizlabs/php_codesniffer": "^3.7",
+ "symfony/browser-kit": "6.2.*",
+ "symfony/phpunit-bridge": "^6.2",
+ "symfony/stopwatch": "6.2.*",
+ "symfony/web-profiler-bundle": "6.2.*"
+ },
+ "config": {
+ "allow-plugins": {
+ "php-http/discovery": true,
+ "symfony/flex": true,
+ "symfony/runtime": true
+ },
+ "sort-packages": true
+ },
+ "autoload": {
+ "psr-4": {
+ "RashinMe\\": "src/"
+ }
+ },
+ "autoload-dev": {
+ "psr-4": {
+ "RashinMe\\": "tests/"
+ }
+ },
+ "replace": {
+ "symfony/polyfill-ctype": "*",
+ "symfony/polyfill-iconv": "*",
+ "symfony/polyfill-php72": "*",
+ "symfony/polyfill-php73": "*",
+ "symfony/polyfill-php74": "*",
+ "symfony/polyfill-php80": "*",
+ "symfony/polyfill-php81": "*"
+ },
+ "scripts": {
+ "auto-scripts": {
+ "cache:clear": "symfony-cmd",
+ "assets:install %PUBLIC_DIR%": "symfony-cmd"
+ },
+ "post-install-cmd": [
+ "@auto-scripts"
+ ],
+ "post-update-cmd": [
+ "@auto-scripts"
+ ]
+ },
+ "conflict": {
+ "symfony/symfony": "*"
+ },
+ "extra": {
+ "symfony": {
+ "allow-contrib": false,
+ "require": "6.2.*"
+ }
+ }
+}
diff --git a/composer.lock b/composer.lock
new file mode 100644
index 0000000..357860d
--- /dev/null
+++ b/composer.lock
@@ -0,0 +1,7453 @@
+{
+ "_readme": [
+ "This file locks the dependencies of your project to a known state",
+ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
+ "This file is @generated automatically"
+ ],
+ "content-hash": "e9349eee1f8a88f56ee9e61a7769b30c",
+ "packages": [
+ {
+ "name": "doctrine/cache",
+ "version": "2.2.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/cache.git",
+ "reference": "1ca8f21980e770095a31456042471a57bc4c68fb"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/cache/zipball/1ca8f21980e770095a31456042471a57bc4c68fb",
+ "reference": "1ca8f21980e770095a31456042471a57bc4c68fb",
+ "shasum": ""
+ },
+ "require": {
+ "php": "~7.1 || ^8.0"
+ },
+ "conflict": {
+ "doctrine/common": ">2.2,<2.4"
+ },
+ "require-dev": {
+ "cache/integration-tests": "dev-master",
+ "doctrine/coding-standard": "^9",
+ "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5",
+ "psr/cache": "^1.0 || ^2.0 || ^3.0",
+ "symfony/cache": "^4.4 || ^5.4 || ^6",
+ "symfony/var-exporter": "^4.4 || ^5.4 || ^6"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Common\\Cache\\": "lib/Doctrine/Common/Cache"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Johannes Schmitt",
+ "email": "schmittjoh@gmail.com"
+ }
+ ],
+ "description": "PHP Doctrine Cache library is a popular cache implementation that supports many different drivers such as redis, memcache, apc, mongodb and others.",
+ "homepage": "https://www.doctrine-project.org/projects/cache.html",
+ "keywords": [
+ "abstraction",
+ "apcu",
+ "cache",
+ "caching",
+ "couchdb",
+ "memcached",
+ "php",
+ "redis",
+ "xcache"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/cache/issues",
+ "source": "https://github.com/doctrine/cache/tree/2.2.0"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fcache",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-05-20T20:07:39+00:00"
+ },
+ {
+ "name": "doctrine/collections",
+ "version": "2.1.2",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/collections.git",
+ "reference": "db8cda536a034337f7dd63febecc713d4957f9ee"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/collections/zipball/db8cda536a034337f7dd63febecc713d4957f9ee",
+ "reference": "db8cda536a034337f7dd63febecc713d4957f9ee",
+ "shasum": ""
+ },
+ "require": {
+ "doctrine/deprecations": "^1",
+ "php": "^8.1"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^10.0",
+ "ext-json": "*",
+ "phpstan/phpstan": "^1.8",
+ "phpstan/phpstan-phpunit": "^1.0",
+ "phpunit/phpunit": "^9.5",
+ "vimeo/psalm": "^4.22"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Common\\Collections\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Johannes Schmitt",
+ "email": "schmittjoh@gmail.com"
+ }
+ ],
+ "description": "PHP Doctrine Collections library that adds additional functionality on top of PHP arrays.",
+ "homepage": "https://www.doctrine-project.org/projects/collections.html",
+ "keywords": [
+ "array",
+ "collections",
+ "iterators",
+ "php"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/collections/issues",
+ "source": "https://github.com/doctrine/collections/tree/2.1.2"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fcollections",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-12-27T23:41:38+00:00"
+ },
+ {
+ "name": "doctrine/common",
+ "version": "3.4.3",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/common.git",
+ "reference": "8b5e5650391f851ed58910b3e3d48a71062eeced"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/common/zipball/8b5e5650391f851ed58910b3e3d48a71062eeced",
+ "reference": "8b5e5650391f851ed58910b3e3d48a71062eeced",
+ "shasum": ""
+ },
+ "require": {
+ "doctrine/persistence": "^2.0 || ^3.0",
+ "php": "^7.1 || ^8.0"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^9.0 || ^10.0",
+ "doctrine/collections": "^1",
+ "phpstan/phpstan": "^1.4.1",
+ "phpstan/phpstan-phpunit": "^1",
+ "phpunit/phpunit": "^7.5.20 || ^8.5 || ^9.0",
+ "squizlabs/php_codesniffer": "^3.0",
+ "symfony/phpunit-bridge": "^6.1",
+ "vimeo/psalm": "^4.4"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Common\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Johannes Schmitt",
+ "email": "schmittjoh@gmail.com"
+ },
+ {
+ "name": "Marco Pivetta",
+ "email": "ocramius@gmail.com"
+ }
+ ],
+ "description": "PHP Doctrine Common project is a library that provides additional functionality that other Doctrine projects depend on such as better reflection support, proxies and much more.",
+ "homepage": "https://www.doctrine-project.org/projects/common.html",
+ "keywords": [
+ "common",
+ "doctrine",
+ "php"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/common/issues",
+ "source": "https://github.com/doctrine/common/tree/3.4.3"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fcommon",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-10-09T11:47:59+00:00"
+ },
+ {
+ "name": "doctrine/dbal",
+ "version": "3.6.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/dbal.git",
+ "reference": "57815c7bbcda3cd18871d253c1dd8cbe56f8526e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/dbal/zipball/57815c7bbcda3cd18871d253c1dd8cbe56f8526e",
+ "reference": "57815c7bbcda3cd18871d253c1dd8cbe56f8526e",
+ "shasum": ""
+ },
+ "require": {
+ "composer-runtime-api": "^2",
+ "doctrine/cache": "^1.11|^2.0",
+ "doctrine/deprecations": "^0.5.3|^1",
+ "doctrine/event-manager": "^1|^2",
+ "php": "^7.4 || ^8.0",
+ "psr/cache": "^1|^2|^3",
+ "psr/log": "^1|^2|^3"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "11.1.0",
+ "fig/log-test": "^1",
+ "jetbrains/phpstorm-stubs": "2022.3",
+ "phpstan/phpstan": "1.10.3",
+ "phpstan/phpstan-strict-rules": "^1.5",
+ "phpunit/phpunit": "9.6.4",
+ "psalm/plugin-phpunit": "0.18.4",
+ "squizlabs/php_codesniffer": "3.7.2",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/console": "^4.4|^5.4|^6.0",
+ "vimeo/psalm": "4.30.0"
+ },
+ "suggest": {
+ "symfony/console": "For helpful console commands such as SQL execution and import of files."
+ },
+ "bin": [
+ "bin/doctrine-dbal"
+ ],
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\DBAL\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ }
+ ],
+ "description": "Powerful PHP database abstraction layer (DBAL) with many features for database schema introspection and management.",
+ "homepage": "https://www.doctrine-project.org/projects/dbal.html",
+ "keywords": [
+ "abstraction",
+ "database",
+ "db2",
+ "dbal",
+ "mariadb",
+ "mssql",
+ "mysql",
+ "oci8",
+ "oracle",
+ "pdo",
+ "pgsql",
+ "postgresql",
+ "queryobject",
+ "sasql",
+ "sql",
+ "sqlite",
+ "sqlserver",
+ "sqlsrv"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/dbal/issues",
+ "source": "https://github.com/doctrine/dbal/tree/3.6.1"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fdbal",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-02T19:26:24+00:00"
+ },
+ {
+ "name": "doctrine/deprecations",
+ "version": "v1.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/deprecations.git",
+ "reference": "0e2a4f1f8cdfc7a92ec3b01c9334898c806b30de"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/deprecations/zipball/0e2a4f1f8cdfc7a92ec3b01c9334898c806b30de",
+ "reference": "0e2a4f1f8cdfc7a92ec3b01c9334898c806b30de",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^7.1|^8.0"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^9",
+ "phpunit/phpunit": "^7.5|^8.5|^9.5",
+ "psr/log": "^1|^2|^3"
+ },
+ "suggest": {
+ "psr/log": "Allows logging deprecations via PSR-3 logger implementation"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Deprecations\\": "lib/Doctrine/Deprecations"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "description": "A small layer on top of trigger_error(E_USER_DEPRECATED) or PSR-3 logging with options to disable all deprecations or selectively for packages.",
+ "homepage": "https://www.doctrine-project.org/",
+ "support": {
+ "issues": "https://github.com/doctrine/deprecations/issues",
+ "source": "https://github.com/doctrine/deprecations/tree/v1.0.0"
+ },
+ "time": "2022-05-02T15:47:09+00:00"
+ },
+ {
+ "name": "doctrine/doctrine-bundle",
+ "version": "2.8.3",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/DoctrineBundle.git",
+ "reference": "fd67ba64db3c806f626a33dcab15a4db0c77652e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/DoctrineBundle/zipball/fd67ba64db3c806f626a33dcab15a4db0c77652e",
+ "reference": "fd67ba64db3c806f626a33dcab15a4db0c77652e",
+ "shasum": ""
+ },
+ "require": {
+ "doctrine/cache": "^1.11 || ^2.0",
+ "doctrine/dbal": "^3.4.0",
+ "doctrine/persistence": "^2.2 || ^3",
+ "doctrine/sql-formatter": "^1.0.1",
+ "php": "^7.4 || ^8.0",
+ "symfony/cache": "^5.4 || ^6.0",
+ "symfony/config": "^5.4 || ^6.0",
+ "symfony/console": "^5.4 || ^6.0",
+ "symfony/dependency-injection": "^5.4 || ^6.0",
+ "symfony/deprecation-contracts": "^2.1 || ^3",
+ "symfony/doctrine-bridge": "^5.4.19 || ^6.0.7",
+ "symfony/framework-bundle": "^5.4 || ^6.0",
+ "symfony/service-contracts": "^1.1.1 || ^2.0 || ^3"
+ },
+ "conflict": {
+ "doctrine/annotations": ">=3.0",
+ "doctrine/orm": "<2.11 || >=3.0",
+ "twig/twig": "<1.34 || >=2.0,<2.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1 || ^2",
+ "doctrine/coding-standard": "^9.0",
+ "doctrine/orm": "^2.11 || ^3.0",
+ "friendsofphp/proxy-manager-lts": "^1.0",
+ "phpunit/phpunit": "^9.5.26 || ^10.0",
+ "psalm/plugin-phpunit": "^0.18.4",
+ "psalm/plugin-symfony": "^4",
+ "psr/log": "^1.1.4 || ^2.0 || ^3.0",
+ "symfony/phpunit-bridge": "^6.1",
+ "symfony/property-info": "^5.4 || ^6.0",
+ "symfony/proxy-manager-bridge": "^5.4 || ^6.0",
+ "symfony/security-bundle": "^5.4 || ^6.0",
+ "symfony/twig-bridge": "^5.4 || ^6.0",
+ "symfony/validator": "^5.4 || ^6.0",
+ "symfony/web-profiler-bundle": "^5.4 || ^6.0",
+ "symfony/yaml": "^5.4 || ^6.0",
+ "twig/twig": "^1.34 || ^2.12 || ^3.0",
+ "vimeo/psalm": "^4.30"
+ },
+ "suggest": {
+ "doctrine/orm": "The Doctrine ORM integration is optional in the bundle.",
+ "ext-pdo": "*",
+ "symfony/web-profiler-bundle": "To use the data collector."
+ },
+ "type": "symfony-bundle",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Bundle\\DoctrineBundle\\": ""
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ },
+ {
+ "name": "Doctrine Project",
+ "homepage": "https://www.doctrine-project.org/"
+ }
+ ],
+ "description": "Symfony DoctrineBundle",
+ "homepage": "https://www.doctrine-project.org",
+ "keywords": [
+ "database",
+ "dbal",
+ "orm",
+ "persistence"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/DoctrineBundle/issues",
+ "source": "https://github.com/doctrine/DoctrineBundle/tree/2.8.3"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fdoctrine-bundle",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-03T09:32:42+00:00"
+ },
+ {
+ "name": "doctrine/doctrine-migrations-bundle",
+ "version": "3.2.2",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/DoctrineMigrationsBundle.git",
+ "reference": "3393f411ba25ade21969c33f2053220044854d01"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/DoctrineMigrationsBundle/zipball/3393f411ba25ade21969c33f2053220044854d01",
+ "reference": "3393f411ba25ade21969c33f2053220044854d01",
+ "shasum": ""
+ },
+ "require": {
+ "doctrine/doctrine-bundle": "~1.0|~2.0",
+ "doctrine/migrations": "^3.2",
+ "php": "^7.2|^8.0",
+ "symfony/framework-bundle": "~3.4|~4.0|~5.0|~6.0"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^8.0",
+ "doctrine/orm": "^2.6",
+ "doctrine/persistence": "^1.3||^2.0",
+ "phpstan/phpstan": "^0.12",
+ "phpstan/phpstan-deprecation-rules": "^0.12",
+ "phpstan/phpstan-phpunit": "^0.12",
+ "phpstan/phpstan-strict-rules": "^0.12",
+ "phpunit/phpunit": "^8.0|^9.0",
+ "vimeo/psalm": "^4.11"
+ },
+ "type": "symfony-bundle",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Bundle\\MigrationsBundle\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Doctrine Project",
+ "homepage": "https://www.doctrine-project.org"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony DoctrineMigrationsBundle",
+ "homepage": "https://www.doctrine-project.org",
+ "keywords": [
+ "dbal",
+ "migrations",
+ "schema"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/DoctrineMigrationsBundle/issues",
+ "source": "https://github.com/doctrine/DoctrineMigrationsBundle/tree/3.2.2"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fdoctrine-migrations-bundle",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-02-01T18:08:07+00:00"
+ },
+ {
+ "name": "doctrine/event-manager",
+ "version": "2.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/event-manager.git",
+ "reference": "750671534e0241a7c50ea5b43f67e23eb5c96f32"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/event-manager/zipball/750671534e0241a7c50ea5b43f67e23eb5c96f32",
+ "reference": "750671534e0241a7c50ea5b43f67e23eb5c96f32",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^8.1"
+ },
+ "conflict": {
+ "doctrine/common": "<2.9"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^10",
+ "phpstan/phpstan": "^1.8.8",
+ "phpunit/phpunit": "^9.5",
+ "vimeo/psalm": "^4.28"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Common\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Johannes Schmitt",
+ "email": "schmittjoh@gmail.com"
+ },
+ {
+ "name": "Marco Pivetta",
+ "email": "ocramius@gmail.com"
+ }
+ ],
+ "description": "The Doctrine Event Manager is a simple PHP event system that was built to be used with the various Doctrine projects.",
+ "homepage": "https://www.doctrine-project.org/projects/event-manager.html",
+ "keywords": [
+ "event",
+ "event dispatcher",
+ "event manager",
+ "event system",
+ "events"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/event-manager/issues",
+ "source": "https://github.com/doctrine/event-manager/tree/2.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fevent-manager",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-10-12T20:59:15+00:00"
+ },
+ {
+ "name": "doctrine/inflector",
+ "version": "2.0.6",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/inflector.git",
+ "reference": "d9d313a36c872fd6ee06d9a6cbcf713eaa40f024"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/inflector/zipball/d9d313a36c872fd6ee06d9a6cbcf713eaa40f024",
+ "reference": "d9d313a36c872fd6ee06d9a6cbcf713eaa40f024",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^7.2 || ^8.0"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^10",
+ "phpstan/phpstan": "^1.8",
+ "phpstan/phpstan-phpunit": "^1.1",
+ "phpstan/phpstan-strict-rules": "^1.3",
+ "phpunit/phpunit": "^8.5 || ^9.5",
+ "vimeo/psalm": "^4.25"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Inflector\\": "lib/Doctrine/Inflector"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Johannes Schmitt",
+ "email": "schmittjoh@gmail.com"
+ }
+ ],
+ "description": "PHP Doctrine Inflector is a small library that can perform string manipulations with regard to upper/lowercase and singular/plural forms of words.",
+ "homepage": "https://www.doctrine-project.org/projects/inflector.html",
+ "keywords": [
+ "inflection",
+ "inflector",
+ "lowercase",
+ "manipulation",
+ "php",
+ "plural",
+ "singular",
+ "strings",
+ "uppercase",
+ "words"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/inflector/issues",
+ "source": "https://github.com/doctrine/inflector/tree/2.0.6"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Finflector",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-10-20T09:10:12+00:00"
+ },
+ {
+ "name": "doctrine/instantiator",
+ "version": "1.5.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/instantiator.git",
+ "reference": "0a0fa9780f5d4e507415a065172d26a98d02047b"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/instantiator/zipball/0a0fa9780f5d4e507415a065172d26a98d02047b",
+ "reference": "0a0fa9780f5d4e507415a065172d26a98d02047b",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^7.1 || ^8.0"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^9 || ^11",
+ "ext-pdo": "*",
+ "ext-phar": "*",
+ "phpbench/phpbench": "^0.16 || ^1",
+ "phpstan/phpstan": "^1.4",
+ "phpstan/phpstan-phpunit": "^1",
+ "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5",
+ "vimeo/psalm": "^4.30 || ^5.4"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Instantiator\\": "src/Doctrine/Instantiator/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Marco Pivetta",
+ "email": "ocramius@gmail.com",
+ "homepage": "https://ocramius.github.io/"
+ }
+ ],
+ "description": "A small, lightweight utility to instantiate objects in PHP without invoking their constructors",
+ "homepage": "https://www.doctrine-project.org/projects/instantiator.html",
+ "keywords": [
+ "constructor",
+ "instantiate"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/instantiator/issues",
+ "source": "https://github.com/doctrine/instantiator/tree/1.5.0"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Finstantiator",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-12-30T00:15:36+00:00"
+ },
+ {
+ "name": "doctrine/lexer",
+ "version": "2.1.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/lexer.git",
+ "reference": "39ab8fcf5a51ce4b85ca97c7a7d033eb12831124"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/lexer/zipball/39ab8fcf5a51ce4b85ca97c7a7d033eb12831124",
+ "reference": "39ab8fcf5a51ce4b85ca97c7a7d033eb12831124",
+ "shasum": ""
+ },
+ "require": {
+ "doctrine/deprecations": "^1.0",
+ "php": "^7.1 || ^8.0"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^9 || ^10",
+ "phpstan/phpstan": "^1.3",
+ "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5",
+ "psalm/plugin-phpunit": "^0.18.3",
+ "vimeo/psalm": "^4.11 || ^5.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Common\\Lexer\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Johannes Schmitt",
+ "email": "schmittjoh@gmail.com"
+ }
+ ],
+ "description": "PHP Doctrine Lexer parser library that can be used in Top-Down, Recursive Descent Parsers.",
+ "homepage": "https://www.doctrine-project.org/projects/lexer.html",
+ "keywords": [
+ "annotations",
+ "docblock",
+ "lexer",
+ "parser",
+ "php"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/lexer/issues",
+ "source": "https://github.com/doctrine/lexer/tree/2.1.0"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Flexer",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-12-14T08:49:07+00:00"
+ },
+ {
+ "name": "doctrine/migrations",
+ "version": "3.6.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/migrations.git",
+ "reference": "e542ad8bcd606d7a18d0875babb8a6d963c9c059"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/migrations/zipball/e542ad8bcd606d7a18d0875babb8a6d963c9c059",
+ "reference": "e542ad8bcd606d7a18d0875babb8a6d963c9c059",
+ "shasum": ""
+ },
+ "require": {
+ "composer-runtime-api": "^2",
+ "doctrine/dbal": "^3.5.1",
+ "doctrine/deprecations": "^0.5.3 || ^1",
+ "doctrine/event-manager": "^1.2 || ^2.0",
+ "php": "^8.1",
+ "psr/log": "^1.1.3 || ^2 || ^3",
+ "symfony/console": "^4.4.16 || ^5.4 || ^6.0",
+ "symfony/stopwatch": "^4.4 || ^5.4 || ^6.0",
+ "symfony/var-exporter": "^6.2"
+ },
+ "conflict": {
+ "doctrine/orm": "<2.12"
+ },
+ "require-dev": {
+ "doctrine/coding-standard": "^9",
+ "doctrine/orm": "^2.13",
+ "doctrine/persistence": "^2 || ^3",
+ "doctrine/sql-formatter": "^1.0",
+ "ext-pdo_sqlite": "*",
+ "phpstan/phpstan": "^1.5",
+ "phpstan/phpstan-deprecation-rules": "^1",
+ "phpstan/phpstan-phpunit": "^1.1",
+ "phpstan/phpstan-strict-rules": "^1.1",
+ "phpstan/phpstan-symfony": "^1.1",
+ "phpunit/phpunit": "^9.5.24",
+ "symfony/cache": "^4.4 || ^5.4 || ^6.0",
+ "symfony/process": "^4.4 || ^5.4 || ^6.0",
+ "symfony/yaml": "^4.4 || ^5.4 || ^6.0"
+ },
+ "suggest": {
+ "doctrine/sql-formatter": "Allows to generate formatted SQL with the diff command.",
+ "symfony/yaml": "Allows the use of yaml for migration configuration files."
+ },
+ "bin": [
+ "bin/doctrine-migrations"
+ ],
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Migrations\\": "lib/Doctrine/Migrations"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Michael Simonson",
+ "email": "contact@mikesimonson.com"
+ }
+ ],
+ "description": "PHP Doctrine Migrations project offer additional functionality on top of the database abstraction layer (DBAL) for versioning your database schema and easily deploying changes to it. It is a very easy to use and a powerful tool.",
+ "homepage": "https://www.doctrine-project.org/projects/migrations.html",
+ "keywords": [
+ "database",
+ "dbal",
+ "migrations"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/migrations/issues",
+ "source": "https://github.com/doctrine/migrations/tree/3.6.0"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fmigrations",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-15T18:49:46+00:00"
+ },
+ {
+ "name": "doctrine/orm",
+ "version": "2.14.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/orm.git",
+ "reference": "de7eee5ed7b1b35c99b118f26f210a8281e6db8e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/orm/zipball/de7eee5ed7b1b35c99b118f26f210a8281e6db8e",
+ "reference": "de7eee5ed7b1b35c99b118f26f210a8281e6db8e",
+ "shasum": ""
+ },
+ "require": {
+ "composer-runtime-api": "^2",
+ "doctrine/cache": "^1.12.1 || ^2.1.1",
+ "doctrine/collections": "^1.5 || ^2.0",
+ "doctrine/common": "^3.0.3",
+ "doctrine/dbal": "^2.13.1 || ^3.2",
+ "doctrine/deprecations": "^0.5.3 || ^1",
+ "doctrine/event-manager": "^1.2 || ^2",
+ "doctrine/inflector": "^1.4 || ^2.0",
+ "doctrine/instantiator": "^1.3",
+ "doctrine/lexer": "^1.2.3 || ^2",
+ "doctrine/persistence": "^2.4 || ^3",
+ "ext-ctype": "*",
+ "php": "^7.1 || ^8.0",
+ "psr/cache": "^1 || ^2 || ^3",
+ "symfony/console": "^4.2 || ^5.0 || ^6.0",
+ "symfony/polyfill-php72": "^1.23",
+ "symfony/polyfill-php80": "^1.16"
+ },
+ "conflict": {
+ "doctrine/annotations": "<1.13 || >= 3.0"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.13 || ^2",
+ "doctrine/coding-standard": "^9.0.2 || ^11.0",
+ "phpbench/phpbench": "^0.16.10 || ^1.0",
+ "phpstan/phpstan": "~1.4.10 || 1.9.8",
+ "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5",
+ "psr/log": "^1 || ^2 || ^3",
+ "squizlabs/php_codesniffer": "3.7.1",
+ "symfony/cache": "^4.4 || ^5.4 || ^6.0",
+ "symfony/var-exporter": "^4.4 || ^5.4 || ^6.2",
+ "symfony/yaml": "^3.4 || ^4.0 || ^5.0 || ^6.0",
+ "vimeo/psalm": "4.30.0 || 5.4.0"
+ },
+ "suggest": {
+ "ext-dom": "Provides support for XSD validation for XML mapping files",
+ "symfony/cache": "Provides cache support for Setup Tool with doctrine/cache 2.0",
+ "symfony/yaml": "If you want to use YAML Metadata Mapping Driver"
+ },
+ "bin": [
+ "bin/doctrine"
+ ],
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\ORM\\": "lib/Doctrine/ORM"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Marco Pivetta",
+ "email": "ocramius@gmail.com"
+ }
+ ],
+ "description": "Object-Relational-Mapper for PHP",
+ "homepage": "https://www.doctrine-project.org/projects/orm.html",
+ "keywords": [
+ "database",
+ "orm"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/orm/issues",
+ "source": "https://github.com/doctrine/orm/tree/2.14.1"
+ },
+ "time": "2023-01-16T18:36:59+00:00"
+ },
+ {
+ "name": "doctrine/persistence",
+ "version": "3.1.4",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/persistence.git",
+ "reference": "8bf8ab15960787f1a49d405f6eb8c787b4841119"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/persistence/zipball/8bf8ab15960787f1a49d405f6eb8c787b4841119",
+ "reference": "8bf8ab15960787f1a49d405f6eb8c787b4841119",
+ "shasum": ""
+ },
+ "require": {
+ "doctrine/event-manager": "^1 || ^2",
+ "php": "^7.2 || ^8.0",
+ "psr/cache": "^1.0 || ^2.0 || ^3.0"
+ },
+ "conflict": {
+ "doctrine/common": "<2.10"
+ },
+ "require-dev": {
+ "composer/package-versions-deprecated": "^1.11",
+ "doctrine/coding-standard": "^11",
+ "doctrine/common": "^3.0",
+ "phpstan/phpstan": "1.9.4",
+ "phpstan/phpstan-phpunit": "^1",
+ "phpstan/phpstan-strict-rules": "^1.1",
+ "phpunit/phpunit": "^8.5 || ^9.5",
+ "symfony/cache": "^4.4 || ^5.4 || ^6.0",
+ "vimeo/psalm": "4.30.0 || 5.3.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\Persistence\\": "src/Persistence"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Guilherme Blanco",
+ "email": "guilhermeblanco@gmail.com"
+ },
+ {
+ "name": "Roman Borschel",
+ "email": "roman@code-factory.org"
+ },
+ {
+ "name": "Benjamin Eberlei",
+ "email": "kontakt@beberlei.de"
+ },
+ {
+ "name": "Jonathan Wage",
+ "email": "jonwage@gmail.com"
+ },
+ {
+ "name": "Johannes Schmitt",
+ "email": "schmittjoh@gmail.com"
+ },
+ {
+ "name": "Marco Pivetta",
+ "email": "ocramius@gmail.com"
+ }
+ ],
+ "description": "The Doctrine Persistence project is a set of shared interfaces and functionality that the different Doctrine object mappers share.",
+ "homepage": "https://www.doctrine-project.org/projects/persistence.html",
+ "keywords": [
+ "mapper",
+ "object",
+ "odm",
+ "orm",
+ "persistence"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/persistence/issues",
+ "source": "https://github.com/doctrine/persistence/tree/3.1.4"
+ },
+ "funding": [
+ {
+ "url": "https://www.doctrine-project.org/sponsorship.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://www.patreon.com/phpdoctrine",
+ "type": "patreon"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/doctrine%2Fpersistence",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-03T11:13:07+00:00"
+ },
+ {
+ "name": "doctrine/sql-formatter",
+ "version": "1.1.3",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/doctrine/sql-formatter.git",
+ "reference": "25a06c7bf4c6b8218f47928654252863ffc890a5"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/doctrine/sql-formatter/zipball/25a06c7bf4c6b8218f47928654252863ffc890a5",
+ "reference": "25a06c7bf4c6b8218f47928654252863ffc890a5",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^7.1 || ^8.0"
+ },
+ "require-dev": {
+ "bamarni/composer-bin-plugin": "^1.4"
+ },
+ "bin": [
+ "bin/sql-formatter"
+ ],
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Doctrine\\SqlFormatter\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Jeremy Dorn",
+ "email": "jeremy@jeremydorn.com",
+ "homepage": "https://jeremydorn.com/"
+ }
+ ],
+ "description": "a PHP SQL highlighting library",
+ "homepage": "https://github.com/doctrine/sql-formatter/",
+ "keywords": [
+ "highlight",
+ "sql"
+ ],
+ "support": {
+ "issues": "https://github.com/doctrine/sql-formatter/issues",
+ "source": "https://github.com/doctrine/sql-formatter/tree/1.1.3"
+ },
+ "time": "2022-05-23T21:33:49+00:00"
+ },
+ {
+ "name": "dompdf/dompdf",
+ "version": "v2.0.3",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/dompdf/dompdf.git",
+ "reference": "e8d2d5e37e8b0b30f0732a011295ab80680d7e85"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/dompdf/dompdf/zipball/e8d2d5e37e8b0b30f0732a011295ab80680d7e85",
+ "reference": "e8d2d5e37e8b0b30f0732a011295ab80680d7e85",
+ "shasum": ""
+ },
+ "require": {
+ "ext-dom": "*",
+ "ext-mbstring": "*",
+ "masterminds/html5": "^2.0",
+ "phenx/php-font-lib": ">=0.5.4 <1.0.0",
+ "phenx/php-svg-lib": ">=0.3.3 <1.0.0",
+ "php": "^7.1 || ^8.0"
+ },
+ "require-dev": {
+ "ext-json": "*",
+ "ext-zip": "*",
+ "mockery/mockery": "^1.3",
+ "phpunit/phpunit": "^7.5 || ^8 || ^9",
+ "squizlabs/php_codesniffer": "^3.5"
+ },
+ "suggest": {
+ "ext-gd": "Needed to process images",
+ "ext-gmagick": "Improves image processing performance",
+ "ext-imagick": "Improves image processing performance",
+ "ext-zlib": "Needed for pdf stream compression"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Dompdf\\": "src/"
+ },
+ "classmap": [
+ "lib/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "LGPL-2.1"
+ ],
+ "authors": [
+ {
+ "name": "The Dompdf Community",
+ "homepage": "https://github.com/dompdf/dompdf/blob/master/AUTHORS.md"
+ }
+ ],
+ "description": "DOMPDF is a CSS 2.1 compliant HTML to PDF converter",
+ "homepage": "https://github.com/dompdf/dompdf",
+ "support": {
+ "issues": "https://github.com/dompdf/dompdf/issues",
+ "source": "https://github.com/dompdf/dompdf/tree/v2.0.3"
+ },
+ "time": "2023-02-07T12:51:48+00:00"
+ },
+ {
+ "name": "masterminds/html5",
+ "version": "2.7.6",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/Masterminds/html5-php.git",
+ "reference": "897eb517a343a2281f11bc5556d6548db7d93947"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/Masterminds/html5-php/zipball/897eb517a343a2281f11bc5556d6548db7d93947",
+ "reference": "897eb517a343a2281f11bc5556d6548db7d93947",
+ "shasum": ""
+ },
+ "require": {
+ "ext-ctype": "*",
+ "ext-dom": "*",
+ "ext-libxml": "*",
+ "php": ">=5.3.0"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^4.8.35 || ^5.7.21 || ^6 || ^7"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "2.7-dev"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Masterminds\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Matt Butcher",
+ "email": "technosophos@gmail.com"
+ },
+ {
+ "name": "Matt Farina",
+ "email": "matt@mattfarina.com"
+ },
+ {
+ "name": "Asmir Mustafic",
+ "email": "goetas@gmail.com"
+ }
+ ],
+ "description": "An HTML5 parser and serializer.",
+ "homepage": "http://masterminds.github.io/html5-php",
+ "keywords": [
+ "HTML5",
+ "dom",
+ "html",
+ "parser",
+ "querypath",
+ "serializer",
+ "xml"
+ ],
+ "support": {
+ "issues": "https://github.com/Masterminds/html5-php/issues",
+ "source": "https://github.com/Masterminds/html5-php/tree/2.7.6"
+ },
+ "time": "2022-08-18T16:18:26+00:00"
+ },
+ {
+ "name": "phenx/php-font-lib",
+ "version": "0.5.4",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/dompdf/php-font-lib.git",
+ "reference": "dd448ad1ce34c63d09baccd05415e361300c35b4"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/dompdf/php-font-lib/zipball/dd448ad1ce34c63d09baccd05415e361300c35b4",
+ "reference": "dd448ad1ce34c63d09baccd05415e361300c35b4",
+ "shasum": ""
+ },
+ "require": {
+ "ext-mbstring": "*"
+ },
+ "require-dev": {
+ "symfony/phpunit-bridge": "^3 || ^4 || ^5"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "FontLib\\": "src/FontLib"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "LGPL-3.0"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Ménager",
+ "email": "fabien.menager@gmail.com"
+ }
+ ],
+ "description": "A library to read, parse, export and make subsets of different types of font files.",
+ "homepage": "https://github.com/PhenX/php-font-lib",
+ "support": {
+ "issues": "https://github.com/dompdf/php-font-lib/issues",
+ "source": "https://github.com/dompdf/php-font-lib/tree/0.5.4"
+ },
+ "time": "2021-12-17T19:44:54+00:00"
+ },
+ {
+ "name": "phenx/php-svg-lib",
+ "version": "0.5.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/dompdf/php-svg-lib.git",
+ "reference": "76876c6cf3080bcb6f249d7d59705108166a6685"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/dompdf/php-svg-lib/zipball/76876c6cf3080bcb6f249d7d59705108166a6685",
+ "reference": "76876c6cf3080bcb6f249d7d59705108166a6685",
+ "shasum": ""
+ },
+ "require": {
+ "ext-mbstring": "*",
+ "php": "^7.1 || ^8.0",
+ "sabberworm/php-css-parser": "^8.4"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Svg\\": "src/Svg"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "LGPL-3.0"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Ménager",
+ "email": "fabien.menager@gmail.com"
+ }
+ ],
+ "description": "A library to read, parse and export to PDF SVG files.",
+ "homepage": "https://github.com/PhenX/php-svg-lib",
+ "support": {
+ "issues": "https://github.com/dompdf/php-svg-lib/issues",
+ "source": "https://github.com/dompdf/php-svg-lib/tree/0.5.0"
+ },
+ "time": "2022-09-06T12:16:56+00:00"
+ },
+ {
+ "name": "psr/cache",
+ "version": "3.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/php-fig/cache.git",
+ "reference": "aa5030cfa5405eccfdcb1083ce040c2cb8d253bf"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/php-fig/cache/zipball/aa5030cfa5405eccfdcb1083ce040c2cb8d253bf",
+ "reference": "aa5030cfa5405eccfdcb1083ce040c2cb8d253bf",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.0.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "1.0.x-dev"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Psr\\Cache\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "PHP-FIG",
+ "homepage": "https://www.php-fig.org/"
+ }
+ ],
+ "description": "Common interface for caching libraries",
+ "keywords": [
+ "cache",
+ "psr",
+ "psr-6"
+ ],
+ "support": {
+ "source": "https://github.com/php-fig/cache/tree/3.0.0"
+ },
+ "time": "2021-02-03T23:26:27+00:00"
+ },
+ {
+ "name": "psr/container",
+ "version": "2.0.2",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/php-fig/container.git",
+ "reference": "c71ecc56dfe541dbd90c5360474fbc405f8d5963"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/php-fig/container/zipball/c71ecc56dfe541dbd90c5360474fbc405f8d5963",
+ "reference": "c71ecc56dfe541dbd90c5360474fbc405f8d5963",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.4.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "2.0.x-dev"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Psr\\Container\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "PHP-FIG",
+ "homepage": "https://www.php-fig.org/"
+ }
+ ],
+ "description": "Common Container Interface (PHP FIG PSR-11)",
+ "homepage": "https://github.com/php-fig/container",
+ "keywords": [
+ "PSR-11",
+ "container",
+ "container-interface",
+ "container-interop",
+ "psr"
+ ],
+ "support": {
+ "issues": "https://github.com/php-fig/container/issues",
+ "source": "https://github.com/php-fig/container/tree/2.0.2"
+ },
+ "time": "2021-11-05T16:47:00+00:00"
+ },
+ {
+ "name": "psr/event-dispatcher",
+ "version": "1.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/php-fig/event-dispatcher.git",
+ "reference": "dbefd12671e8a14ec7f180cab83036ed26714bb0"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/php-fig/event-dispatcher/zipball/dbefd12671e8a14ec7f180cab83036ed26714bb0",
+ "reference": "dbefd12671e8a14ec7f180cab83036ed26714bb0",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.2.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "1.0.x-dev"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Psr\\EventDispatcher\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "PHP-FIG",
+ "homepage": "http://www.php-fig.org/"
+ }
+ ],
+ "description": "Standard interfaces for event handling.",
+ "keywords": [
+ "events",
+ "psr",
+ "psr-14"
+ ],
+ "support": {
+ "issues": "https://github.com/php-fig/event-dispatcher/issues",
+ "source": "https://github.com/php-fig/event-dispatcher/tree/1.0.0"
+ },
+ "time": "2019-01-08T18:20:26+00:00"
+ },
+ {
+ "name": "psr/log",
+ "version": "3.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/php-fig/log.git",
+ "reference": "fe5ea303b0887d5caefd3d431c3e61ad47037001"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/php-fig/log/zipball/fe5ea303b0887d5caefd3d431c3e61ad47037001",
+ "reference": "fe5ea303b0887d5caefd3d431c3e61ad47037001",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.0.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "3.x-dev"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Psr\\Log\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "PHP-FIG",
+ "homepage": "https://www.php-fig.org/"
+ }
+ ],
+ "description": "Common interface for logging libraries",
+ "homepage": "https://github.com/php-fig/log",
+ "keywords": [
+ "log",
+ "psr",
+ "psr-3"
+ ],
+ "support": {
+ "source": "https://github.com/php-fig/log/tree/3.0.0"
+ },
+ "time": "2021-07-14T16:46:02+00:00"
+ },
+ {
+ "name": "sabberworm/php-css-parser",
+ "version": "8.4.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sabberworm/PHP-CSS-Parser.git",
+ "reference": "e41d2140031d533348b2192a83f02d8dd8a71d30"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sabberworm/PHP-CSS-Parser/zipball/e41d2140031d533348b2192a83f02d8dd8a71d30",
+ "reference": "e41d2140031d533348b2192a83f02d8dd8a71d30",
+ "shasum": ""
+ },
+ "require": {
+ "ext-iconv": "*",
+ "php": ">=5.6.20"
+ },
+ "require-dev": {
+ "codacy/coverage": "^1.4",
+ "phpunit/phpunit": "^4.8.36"
+ },
+ "suggest": {
+ "ext-mbstring": "for parsing UTF-8 CSS"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Sabberworm\\CSS\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Raphael Schweikert"
+ }
+ ],
+ "description": "Parser for CSS Files written in PHP",
+ "homepage": "https://www.sabberworm.com/blog/2010/6/10/php-css-parser",
+ "keywords": [
+ "css",
+ "parser",
+ "stylesheet"
+ ],
+ "support": {
+ "issues": "https://github.com/sabberworm/PHP-CSS-Parser/issues",
+ "source": "https://github.com/sabberworm/PHP-CSS-Parser/tree/8.4.0"
+ },
+ "time": "2021-12-11T13:40:54+00:00"
+ },
+ {
+ "name": "ser/dto-request-bundle",
+ "version": "0.1.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/SerRashin/dto-request-bundle.git",
+ "reference": "ded02e67b2a3fbe26539d73521c548a4c3a73637"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/SerRashin/dto-request-bundle/zipball/ded02e67b2a3fbe26539d73521c548a4c3a73637",
+ "reference": "ded02e67b2a3fbe26539d73521c548a4c3a73637",
+ "shasum": ""
+ },
+ "require": {
+ "ext-json": "*",
+ "php": "^8.1",
+ "symfony/dependency-injection": "^6.2",
+ "symfony/http-kernel": "^6.2"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0",
+ "squizlabs/php_codesniffer": "^3.7"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Ser\\DTORequestBundle\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Sergey Rashin",
+ "email": "ser@rashin.me"
+ }
+ ],
+ "description": "Convert Symfony request to DTO object.",
+ "keywords": [
+ "dto",
+ "request",
+ "symfony"
+ ],
+ "support": {
+ "issues": "https://github.com/SerRashin/dto-request-bundle/issues",
+ "source": "https://github.com/SerRashin/dto-request-bundle/tree/0.1.0"
+ },
+ "time": "2023-04-10T01:19:42+00:00"
+ },
+ {
+ "name": "symfony/cache",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/cache.git",
+ "reference": "01a36b32f930018764bcbde006fbbe421fa6b61e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/cache/zipball/01a36b32f930018764bcbde006fbbe421fa6b61e",
+ "reference": "01a36b32f930018764bcbde006fbbe421fa6b61e",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "psr/cache": "^2.0|^3.0",
+ "psr/log": "^1.1|^2|^3",
+ "symfony/cache-contracts": "^1.1.7|^2|^3",
+ "symfony/service-contracts": "^1.1|^2|^3",
+ "symfony/var-exporter": "^6.2.7"
+ },
+ "conflict": {
+ "doctrine/dbal": "<2.13.1",
+ "symfony/dependency-injection": "<5.4",
+ "symfony/http-kernel": "<5.4",
+ "symfony/var-dumper": "<5.4"
+ },
+ "provide": {
+ "psr/cache-implementation": "2.0|3.0",
+ "psr/simple-cache-implementation": "1.0|2.0|3.0",
+ "symfony/cache-implementation": "1.1|2.0|3.0"
+ },
+ "require-dev": {
+ "cache/integration-tests": "dev-master",
+ "doctrine/dbal": "^2.13.1|^3.0",
+ "predis/predis": "^1.1",
+ "psr/simple-cache": "^1.0|^2.0|^3.0",
+ "symfony/config": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/filesystem": "^5.4|^6.0",
+ "symfony/http-kernel": "^5.4|^6.0",
+ "symfony/messenger": "^5.4|^6.0",
+ "symfony/var-dumper": "^5.4|^6.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Cache\\": ""
+ },
+ "classmap": [
+ "Traits/ValueWrapper.php"
+ ],
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides extended PSR-6, PSR-16 (and tags) implementations",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "caching",
+ "psr6"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/cache/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-21T16:15:44+00:00"
+ },
+ {
+ "name": "symfony/cache-contracts",
+ "version": "v3.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/cache-contracts.git",
+ "reference": "eeb71f04b6f7f34ca6d15633df82e014528b1632"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/cache-contracts/zipball/eeb71f04b6f7f34ca6d15633df82e014528b1632",
+ "reference": "eeb71f04b6f7f34ca6d15633df82e014528b1632",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "psr/cache": "^3.0"
+ },
+ "suggest": {
+ "symfony/cache-implementation": ""
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.3-dev"
+ },
+ "thanks": {
+ "name": "symfony/contracts",
+ "url": "https://github.com/symfony/contracts"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Contracts\\Cache\\": ""
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Generic abstractions related to caching",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "abstractions",
+ "contracts",
+ "decoupling",
+ "interfaces",
+ "interoperability",
+ "standards"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/cache-contracts/tree/v3.2.1"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-01T10:32:47+00:00"
+ },
+ {
+ "name": "symfony/config",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/config.git",
+ "reference": "249271da6f545d6579e0663374f8249a80be2893"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/config/zipball/249271da6f545d6579e0663374f8249a80be2893",
+ "reference": "249271da6f545d6579e0663374f8249a80be2893",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/filesystem": "^5.4|^6.0",
+ "symfony/polyfill-ctype": "~1.8"
+ },
+ "conflict": {
+ "symfony/finder": "<5.4"
+ },
+ "require-dev": {
+ "symfony/event-dispatcher": "^5.4|^6.0",
+ "symfony/finder": "^5.4|^6.0",
+ "symfony/messenger": "^5.4|^6.0",
+ "symfony/service-contracts": "^1.1|^2|^3",
+ "symfony/yaml": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/yaml": "To use the yaml reference dumper"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Config\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Helps you find, load, combine, autofill and validate configuration values of any kind",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/config/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/console",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/console.git",
+ "reference": "cbad09eb8925b6ad4fb721c7a179344dc4a19d45"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/console/zipball/cbad09eb8925b6ad4fb721c7a179344dc4a19d45",
+ "reference": "cbad09eb8925b6ad4fb721c7a179344dc4a19d45",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/polyfill-mbstring": "~1.0",
+ "symfony/service-contracts": "^1.1|^2|^3",
+ "symfony/string": "^5.4|^6.0"
+ },
+ "conflict": {
+ "symfony/dependency-injection": "<5.4",
+ "symfony/dotenv": "<5.4",
+ "symfony/event-dispatcher": "<5.4",
+ "symfony/lock": "<5.4",
+ "symfony/process": "<5.4"
+ },
+ "provide": {
+ "psr/log-implementation": "1.0|2.0|3.0"
+ },
+ "require-dev": {
+ "psr/log": "^1|^2|^3",
+ "symfony/config": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/event-dispatcher": "^5.4|^6.0",
+ "symfony/lock": "^5.4|^6.0",
+ "symfony/process": "^5.4|^6.0",
+ "symfony/var-dumper": "^5.4|^6.0"
+ },
+ "suggest": {
+ "psr/log": "For using the console logger",
+ "symfony/event-dispatcher": "",
+ "symfony/lock": "",
+ "symfony/process": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Console\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Eases the creation of beautiful and testable command line interfaces",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "cli",
+ "command line",
+ "console",
+ "terminal"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/console/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-25T17:00:03+00:00"
+ },
+ {
+ "name": "symfony/dependency-injection",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/dependency-injection.git",
+ "reference": "83369dd4ec84bba9673524d25b79dfbde9e6e84c"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/dependency-injection/zipball/83369dd4ec84bba9673524d25b79dfbde9e6e84c",
+ "reference": "83369dd4ec84bba9673524d25b79dfbde9e6e84c",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "psr/container": "^1.1|^2.0",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/service-contracts": "^1.1.6|^2.0|^3.0",
+ "symfony/var-exporter": "^6.2.7"
+ },
+ "conflict": {
+ "ext-psr": "<1.1|>=2",
+ "symfony/config": "<6.1",
+ "symfony/finder": "<5.4",
+ "symfony/proxy-manager-bridge": "<6.2",
+ "symfony/yaml": "<5.4"
+ },
+ "provide": {
+ "psr/container-implementation": "1.1|2.0",
+ "symfony/service-implementation": "1.1|2.0|3.0"
+ },
+ "require-dev": {
+ "symfony/config": "^6.1",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/yaml": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/config": "",
+ "symfony/expression-language": "For using expressions in service container configuration",
+ "symfony/finder": "For using double-star glob patterns or when GLOB_BRACE portability is required",
+ "symfony/yaml": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\DependencyInjection\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Allows you to standardize and centralize the way objects are constructed in your application",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/dependency-injection/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-16T14:11:02+00:00"
+ },
+ {
+ "name": "symfony/deprecation-contracts",
+ "version": "v3.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/deprecation-contracts.git",
+ "reference": "e2d1534420bd723d0ef5aec58a22c5fe60ce6f5e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/e2d1534420bd723d0ef5aec58a22c5fe60ce6f5e",
+ "reference": "e2d1534420bd723d0ef5aec58a22c5fe60ce6f5e",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.3-dev"
+ },
+ "thanks": {
+ "name": "symfony/contracts",
+ "url": "https://github.com/symfony/contracts"
+ }
+ },
+ "autoload": {
+ "files": [
+ "function.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "A generic function and convention to trigger deprecation notices",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/deprecation-contracts/tree/v3.2.1"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-01T10:25:55+00:00"
+ },
+ {
+ "name": "symfony/doctrine-bridge",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/doctrine-bridge.git",
+ "reference": "35cb5045e15bf6bd89fd1353d9b03ff61dc1feaf"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/doctrine-bridge/zipball/35cb5045e15bf6bd89fd1353d9b03ff61dc1feaf",
+ "reference": "35cb5045e15bf6bd89fd1353d9b03ff61dc1feaf",
+ "shasum": ""
+ },
+ "require": {
+ "doctrine/event-manager": "^1.2|^2",
+ "doctrine/persistence": "^2|^3",
+ "php": ">=8.1",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/polyfill-ctype": "~1.8",
+ "symfony/polyfill-mbstring": "~1.0",
+ "symfony/service-contracts": "^1.1|^2|^3"
+ },
+ "conflict": {
+ "doctrine/dbal": "<2.13.1",
+ "doctrine/lexer": "<1.1",
+ "doctrine/orm": "<2.7.4",
+ "phpunit/phpunit": "<5.4.3",
+ "symfony/cache": "<5.4",
+ "symfony/dependency-injection": "<5.4",
+ "symfony/form": "<5.4.21|>=6,<6.2.7",
+ "symfony/http-kernel": "<6.2",
+ "symfony/messenger": "<5.4",
+ "symfony/property-info": "<5.4",
+ "symfony/security-bundle": "<5.4",
+ "symfony/security-core": "<6.0",
+ "symfony/validator": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.10.4|^2",
+ "doctrine/collections": "^1.0|^2.0",
+ "doctrine/data-fixtures": "^1.1",
+ "doctrine/dbal": "^2.13.1|^3.0",
+ "doctrine/orm": "^2.7.4",
+ "psr/log": "^1|^2|^3",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/config": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/doctrine-messenger": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/form": "^5.4.21|^6.2.7",
+ "symfony/http-kernel": "^6.2",
+ "symfony/messenger": "^5.4|^6.0",
+ "symfony/property-access": "^5.4|^6.0",
+ "symfony/property-info": "^5.4|^6.0",
+ "symfony/proxy-manager-bridge": "^5.4|^6.0",
+ "symfony/security-core": "^6.0",
+ "symfony/stopwatch": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/uid": "^5.4|^6.0",
+ "symfony/validator": "^5.4|^6.0",
+ "symfony/var-dumper": "^5.4|^6.0"
+ },
+ "suggest": {
+ "doctrine/data-fixtures": "",
+ "doctrine/dbal": "",
+ "doctrine/orm": "",
+ "symfony/form": "",
+ "symfony/property-info": "",
+ "symfony/validator": ""
+ },
+ "type": "symfony-bridge",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Bridge\\Doctrine\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides integration for Doctrine with various Symfony components",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/doctrine-bridge/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/dotenv",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/dotenv.git",
+ "reference": "f2b09b7ee21458779df000bd24020b6e9955b393"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/dotenv/zipball/f2b09b7ee21458779df000bd24020b6e9955b393",
+ "reference": "f2b09b7ee21458779df000bd24020b6e9955b393",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "conflict": {
+ "symfony/console": "<5.4",
+ "symfony/process": "<5.4"
+ },
+ "require-dev": {
+ "symfony/console": "^5.4|^6.0",
+ "symfony/process": "^5.4|^6.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Dotenv\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Registers environment variables from a .env file",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "dotenv",
+ "env",
+ "environment"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/dotenv/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/error-handler",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/error-handler.git",
+ "reference": "61e90f94eb014054000bc902257d2763fac09166"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/error-handler/zipball/61e90f94eb014054000bc902257d2763fac09166",
+ "reference": "61e90f94eb014054000bc902257d2763fac09166",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "psr/log": "^1|^2|^3",
+ "symfony/var-dumper": "^5.4|^6.0"
+ },
+ "require-dev": {
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/http-kernel": "^5.4|^6.0",
+ "symfony/serializer": "^5.4|^6.0"
+ },
+ "bin": [
+ "Resources/bin/patch-type-declarations"
+ ],
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\ErrorHandler\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides tools to manage errors and ease debugging PHP code",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/error-handler/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/event-dispatcher",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/event-dispatcher.git",
+ "reference": "404b307de426c1c488e5afad64403e5f145e82a5"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/404b307de426c1c488e5afad64403e5f145e82a5",
+ "reference": "404b307de426c1c488e5afad64403e5f145e82a5",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/event-dispatcher-contracts": "^2|^3"
+ },
+ "conflict": {
+ "symfony/dependency-injection": "<5.4"
+ },
+ "provide": {
+ "psr/event-dispatcher-implementation": "1.0",
+ "symfony/event-dispatcher-implementation": "2.0|3.0"
+ },
+ "require-dev": {
+ "psr/log": "^1|^2|^3",
+ "symfony/config": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/error-handler": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/service-contracts": "^1.1|^2|^3",
+ "symfony/stopwatch": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/dependency-injection": "",
+ "symfony/http-kernel": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\EventDispatcher\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides tools that allow your application components to communicate with each other by dispatching events and listening to them",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/event-dispatcher/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/event-dispatcher-contracts",
+ "version": "v3.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/event-dispatcher-contracts.git",
+ "reference": "0ad3b6f1e4e2da5690fefe075cd53a238646d8dd"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/event-dispatcher-contracts/zipball/0ad3b6f1e4e2da5690fefe075cd53a238646d8dd",
+ "reference": "0ad3b6f1e4e2da5690fefe075cd53a238646d8dd",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "psr/event-dispatcher": "^1"
+ },
+ "suggest": {
+ "symfony/event-dispatcher-implementation": ""
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.3-dev"
+ },
+ "thanks": {
+ "name": "symfony/contracts",
+ "url": "https://github.com/symfony/contracts"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Contracts\\EventDispatcher\\": ""
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Generic abstractions related to dispatching event",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "abstractions",
+ "contracts",
+ "decoupling",
+ "interfaces",
+ "interoperability",
+ "standards"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/event-dispatcher-contracts/tree/v3.2.1"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-01T10:32:47+00:00"
+ },
+ {
+ "name": "symfony/filesystem",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/filesystem.git",
+ "reference": "82b6c62b959f642d000456f08c6d219d749215b3"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/filesystem/zipball/82b6c62b959f642d000456f08c6d219d749215b3",
+ "reference": "82b6c62b959f642d000456f08c6d219d749215b3",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/polyfill-ctype": "~1.8",
+ "symfony/polyfill-mbstring": "~1.8"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Filesystem\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides basic utilities for the filesystem",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/filesystem/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/finder",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/finder.git",
+ "reference": "20808dc6631aecafbe67c186af5dcb370be3a0eb"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/finder/zipball/20808dc6631aecafbe67c186af5dcb370be3a0eb",
+ "reference": "20808dc6631aecafbe67c186af5dcb370be3a0eb",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "symfony/filesystem": "^6.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Finder\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Finds files and directories via an intuitive fluent interface",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/finder/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-16T09:57:23+00:00"
+ },
+ {
+ "name": "symfony/flex",
+ "version": "v2.2.5",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/flex.git",
+ "reference": "2ff8465e7172790a47ab3c129f2b514eb2d8a286"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/flex/zipball/2ff8465e7172790a47ab3c129f2b514eb2d8a286",
+ "reference": "2ff8465e7172790a47ab3c129f2b514eb2d8a286",
+ "shasum": ""
+ },
+ "require": {
+ "composer-plugin-api": "^2.1",
+ "php": ">=8.0"
+ },
+ "require-dev": {
+ "composer/composer": "^2.1",
+ "symfony/dotenv": "^5.4|^6.0",
+ "symfony/filesystem": "^5.4|^6.0",
+ "symfony/phpunit-bridge": "^5.4|^6.0",
+ "symfony/process": "^5.4|^6.0"
+ },
+ "type": "composer-plugin",
+ "extra": {
+ "class": "Symfony\\Flex\\Flex"
+ },
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Flex\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien.potencier@gmail.com"
+ }
+ ],
+ "description": "Composer plugin for Symfony",
+ "support": {
+ "issues": "https://github.com/symfony/flex/issues",
+ "source": "https://github.com/symfony/flex/tree/v2.2.5"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-18T08:03:15+00:00"
+ },
+ {
+ "name": "symfony/framework-bundle",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/framework-bundle.git",
+ "reference": "01b1caa34ae121a192580acd38f66b7cb8b9ecce"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/framework-bundle/zipball/01b1caa34ae121a192580acd38f66b7cb8b9ecce",
+ "reference": "01b1caa34ae121a192580acd38f66b7cb8b9ecce",
+ "shasum": ""
+ },
+ "require": {
+ "composer-runtime-api": ">=2.1",
+ "ext-xml": "*",
+ "php": ">=8.1",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/config": "^6.1",
+ "symfony/dependency-injection": "^6.2",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/error-handler": "^6.1",
+ "symfony/event-dispatcher": "^5.4|^6.0",
+ "symfony/filesystem": "^5.4|^6.0",
+ "symfony/finder": "^5.4|^6.0",
+ "symfony/http-foundation": "^6.2",
+ "symfony/http-kernel": "^6.2.1",
+ "symfony/polyfill-mbstring": "~1.0",
+ "symfony/routing": "^5.4|^6.0"
+ },
+ "conflict": {
+ "doctrine/annotations": "<1.13.1",
+ "doctrine/persistence": "<1.3",
+ "phpdocumentor/reflection-docblock": "<3.2.2",
+ "phpdocumentor/type-resolver": "<1.4.0",
+ "phpunit/phpunit": "<5.4.3",
+ "symfony/asset": "<5.4",
+ "symfony/console": "<5.4",
+ "symfony/dom-crawler": "<5.4",
+ "symfony/dotenv": "<5.4",
+ "symfony/form": "<5.4",
+ "symfony/http-client": "<5.4",
+ "symfony/lock": "<5.4",
+ "symfony/mailer": "<5.4",
+ "symfony/messenger": "<6.2",
+ "symfony/mime": "<6.2",
+ "symfony/property-access": "<5.4",
+ "symfony/property-info": "<5.4",
+ "symfony/security-core": "<5.4",
+ "symfony/security-csrf": "<5.4",
+ "symfony/serializer": "<6.1",
+ "symfony/stopwatch": "<5.4",
+ "symfony/translation": "<5.4",
+ "symfony/twig-bridge": "<5.4",
+ "symfony/twig-bundle": "<5.4",
+ "symfony/validator": "<5.4",
+ "symfony/web-profiler-bundle": "<5.4",
+ "symfony/workflow": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.13.1|^2",
+ "doctrine/persistence": "^1.3|^2|^3",
+ "phpdocumentor/reflection-docblock": "^3.0|^4.0|^5.0",
+ "symfony/asset": "^5.4|^6.0",
+ "symfony/browser-kit": "^5.4|^6.0",
+ "symfony/console": "^5.4.9|^6.0.9",
+ "symfony/css-selector": "^5.4|^6.0",
+ "symfony/dom-crawler": "^5.4|^6.0",
+ "symfony/dotenv": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/form": "^5.4|^6.0",
+ "symfony/html-sanitizer": "^6.1",
+ "symfony/http-client": "^5.4|^6.0",
+ "symfony/lock": "^5.4|^6.0",
+ "symfony/mailer": "^5.4|^6.0",
+ "symfony/messenger": "^6.2",
+ "symfony/mime": "^6.2",
+ "symfony/notifier": "^5.4|^6.0",
+ "symfony/polyfill-intl-icu": "~1.0",
+ "symfony/process": "^5.4|^6.0",
+ "symfony/property-info": "^5.4|^6.0",
+ "symfony/rate-limiter": "^5.4|^6.0",
+ "symfony/security-bundle": "^5.4|^6.0",
+ "symfony/semaphore": "^5.4|^6.0",
+ "symfony/serializer": "^6.1",
+ "symfony/stopwatch": "^5.4|^6.0",
+ "symfony/string": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/twig-bundle": "^5.4|^6.0",
+ "symfony/uid": "^5.4|^6.0",
+ "symfony/validator": "^5.4|^6.0",
+ "symfony/web-link": "^5.4|^6.0",
+ "symfony/workflow": "^5.4|^6.0",
+ "symfony/yaml": "^5.4|^6.0",
+ "twig/twig": "^2.10|^3.0"
+ },
+ "suggest": {
+ "ext-apcu": "For best performance of the system caches",
+ "symfony/console": "For using the console commands",
+ "symfony/form": "For using forms",
+ "symfony/property-info": "For using the property_info service",
+ "symfony/serializer": "For using the serializer service",
+ "symfony/validator": "For using validation",
+ "symfony/web-link": "For using web links, features such as preloading, prefetching or prerendering",
+ "symfony/yaml": "For using the debug:config and lint:yaml commands"
+ },
+ "type": "symfony-bundle",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Bundle\\FrameworkBundle\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides a tight integration between Symfony components and the Symfony full-stack framework",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/framework-bundle/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-24T10:42:00+00:00"
+ },
+ {
+ "name": "symfony/http-foundation",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/http-foundation.git",
+ "reference": "5fc3038d4a594223f9ea42e4e985548f3fcc9a3b"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/http-foundation/zipball/5fc3038d4a594223f9ea42e4e985548f3fcc9a3b",
+ "reference": "5fc3038d4a594223f9ea42e4e985548f3fcc9a3b",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/polyfill-mbstring": "~1.1"
+ },
+ "conflict": {
+ "symfony/cache": "<6.2"
+ },
+ "require-dev": {
+ "predis/predis": "~1.0",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/http-kernel": "^5.4.12|^6.0.12|^6.1.4",
+ "symfony/mime": "^5.4|^6.0",
+ "symfony/rate-limiter": "^5.2|^6.0"
+ },
+ "suggest": {
+ "symfony/mime": "To use the file extension guesser"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\HttpFoundation\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Defines an object-oriented layer for the HTTP specification",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/http-foundation/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-21T10:54:55+00:00"
+ },
+ {
+ "name": "symfony/http-kernel",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/http-kernel.git",
+ "reference": "ca0680ad1e2d678536cc20e0ae33f9e4e5d2becd"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/http-kernel/zipball/ca0680ad1e2d678536cc20e0ae33f9e4e5d2becd",
+ "reference": "ca0680ad1e2d678536cc20e0ae33f9e4e5d2becd",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "psr/log": "^1|^2|^3",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/error-handler": "^6.1",
+ "symfony/event-dispatcher": "^5.4|^6.0",
+ "symfony/http-foundation": "^5.4.21|^6.2.7",
+ "symfony/polyfill-ctype": "^1.8"
+ },
+ "conflict": {
+ "symfony/browser-kit": "<5.4",
+ "symfony/cache": "<5.4",
+ "symfony/config": "<6.1",
+ "symfony/console": "<5.4",
+ "symfony/dependency-injection": "<6.2",
+ "symfony/doctrine-bridge": "<5.4",
+ "symfony/form": "<5.4",
+ "symfony/http-client": "<5.4",
+ "symfony/mailer": "<5.4",
+ "symfony/messenger": "<5.4",
+ "symfony/translation": "<5.4",
+ "symfony/twig-bridge": "<5.4",
+ "symfony/validator": "<5.4",
+ "twig/twig": "<2.13"
+ },
+ "provide": {
+ "psr/log-implementation": "1.0|2.0|3.0"
+ },
+ "require-dev": {
+ "psr/cache": "^1.0|^2.0|^3.0",
+ "symfony/browser-kit": "^5.4|^6.0",
+ "symfony/config": "^6.1",
+ "symfony/console": "^5.4|^6.0",
+ "symfony/css-selector": "^5.4|^6.0",
+ "symfony/dependency-injection": "^6.2",
+ "symfony/dom-crawler": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/finder": "^5.4|^6.0",
+ "symfony/http-client-contracts": "^1.1|^2|^3",
+ "symfony/process": "^5.4|^6.0",
+ "symfony/routing": "^5.4|^6.0",
+ "symfony/stopwatch": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/translation-contracts": "^1.1|^2|^3",
+ "symfony/uid": "^5.4|^6.0",
+ "twig/twig": "^2.13|^3.0.4"
+ },
+ "suggest": {
+ "symfony/browser-kit": "",
+ "symfony/config": "",
+ "symfony/console": "",
+ "symfony/dependency-injection": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\HttpKernel\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides a structured process for converting a Request into a Response",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/http-kernel/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-28T13:26:41+00:00"
+ },
+ {
+ "name": "symfony/mime",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/mime.git",
+ "reference": "62e341f80699badb0ad70b31149c8df89a2d778e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/mime/zipball/62e341f80699badb0ad70b31149c8df89a2d778e",
+ "reference": "62e341f80699badb0ad70b31149c8df89a2d778e",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/polyfill-intl-idn": "^1.10",
+ "symfony/polyfill-mbstring": "^1.0"
+ },
+ "conflict": {
+ "egulias/email-validator": "~3.0.0",
+ "phpdocumentor/reflection-docblock": "<3.2.2",
+ "phpdocumentor/type-resolver": "<1.4.0",
+ "symfony/mailer": "<5.4",
+ "symfony/serializer": "<6.2"
+ },
+ "require-dev": {
+ "egulias/email-validator": "^2.1.10|^3.1|^4",
+ "league/html-to-markdown": "^5.0",
+ "phpdocumentor/reflection-docblock": "^3.0|^4.0|^5.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/property-access": "^5.4|^6.0",
+ "symfony/property-info": "^5.4|^6.0",
+ "symfony/serializer": "^6.2"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Mime\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Allows manipulating MIME messages",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "mime",
+ "mime-type"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/mime/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-24T10:42:00+00:00"
+ },
+ {
+ "name": "symfony/password-hasher",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/password-hasher.git",
+ "reference": "67820d8570bf1c2c2cd87cb76d9d12a9d52ab808"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/password-hasher/zipball/67820d8570bf1c2c2cd87cb76d9d12a9d52ab808",
+ "reference": "67820d8570bf1c2c2cd87cb76d9d12a9d52ab808",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "conflict": {
+ "symfony/security-core": "<5.4"
+ },
+ "require-dev": {
+ "symfony/console": "^5.4|^6.0",
+ "symfony/security-core": "^5.4|^6.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\PasswordHasher\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Robin Chalas",
+ "email": "robin.chalas@gmail.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides password hashing utilities",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "hashing",
+ "password"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/password-hasher/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/polyfill-intl-grapheme",
+ "version": "v1.27.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/polyfill-intl-grapheme.git",
+ "reference": "511a08c03c1960e08a883f4cffcacd219b758354"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/polyfill-intl-grapheme/zipball/511a08c03c1960e08a883f4cffcacd219b758354",
+ "reference": "511a08c03c1960e08a883f4cffcacd219b758354",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.1"
+ },
+ "suggest": {
+ "ext-intl": "For best performance"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "1.27-dev"
+ },
+ "thanks": {
+ "name": "symfony/polyfill",
+ "url": "https://github.com/symfony/polyfill"
+ }
+ },
+ "autoload": {
+ "files": [
+ "bootstrap.php"
+ ],
+ "psr-4": {
+ "Symfony\\Polyfill\\Intl\\Grapheme\\": ""
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony polyfill for intl's grapheme_* functions",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "compatibility",
+ "grapheme",
+ "intl",
+ "polyfill",
+ "portable",
+ "shim"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/polyfill-intl-grapheme/tree/v1.27.0"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-11-03T14:55:06+00:00"
+ },
+ {
+ "name": "symfony/polyfill-intl-idn",
+ "version": "v1.27.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/polyfill-intl-idn.git",
+ "reference": "639084e360537a19f9ee352433b84ce831f3d2da"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/polyfill-intl-idn/zipball/639084e360537a19f9ee352433b84ce831f3d2da",
+ "reference": "639084e360537a19f9ee352433b84ce831f3d2da",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.1",
+ "symfony/polyfill-intl-normalizer": "^1.10",
+ "symfony/polyfill-php72": "^1.10"
+ },
+ "suggest": {
+ "ext-intl": "For best performance"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "1.27-dev"
+ },
+ "thanks": {
+ "name": "symfony/polyfill",
+ "url": "https://github.com/symfony/polyfill"
+ }
+ },
+ "autoload": {
+ "files": [
+ "bootstrap.php"
+ ],
+ "psr-4": {
+ "Symfony\\Polyfill\\Intl\\Idn\\": ""
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Laurent Bassin",
+ "email": "laurent@bassin.info"
+ },
+ {
+ "name": "Trevor Rowbotham",
+ "email": "trevor.rowbotham@pm.me"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony polyfill for intl's idn_to_ascii and idn_to_utf8 functions",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "compatibility",
+ "idn",
+ "intl",
+ "polyfill",
+ "portable",
+ "shim"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/polyfill-intl-idn/tree/v1.27.0"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-11-03T14:55:06+00:00"
+ },
+ {
+ "name": "symfony/polyfill-intl-normalizer",
+ "version": "v1.27.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/polyfill-intl-normalizer.git",
+ "reference": "19bd1e4fcd5b91116f14d8533c57831ed00571b6"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/polyfill-intl-normalizer/zipball/19bd1e4fcd5b91116f14d8533c57831ed00571b6",
+ "reference": "19bd1e4fcd5b91116f14d8533c57831ed00571b6",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.1"
+ },
+ "suggest": {
+ "ext-intl": "For best performance"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "1.27-dev"
+ },
+ "thanks": {
+ "name": "symfony/polyfill",
+ "url": "https://github.com/symfony/polyfill"
+ }
+ },
+ "autoload": {
+ "files": [
+ "bootstrap.php"
+ ],
+ "psr-4": {
+ "Symfony\\Polyfill\\Intl\\Normalizer\\": ""
+ },
+ "classmap": [
+ "Resources/stubs"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony polyfill for intl's Normalizer class and related functions",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "compatibility",
+ "intl",
+ "normalizer",
+ "polyfill",
+ "portable",
+ "shim"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/polyfill-intl-normalizer/tree/v1.27.0"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-11-03T14:55:06+00:00"
+ },
+ {
+ "name": "symfony/polyfill-mbstring",
+ "version": "v1.27.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/polyfill-mbstring.git",
+ "reference": "8ad114f6b39e2c98a8b0e3bd907732c207c2b534"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/polyfill-mbstring/zipball/8ad114f6b39e2c98a8b0e3bd907732c207c2b534",
+ "reference": "8ad114f6b39e2c98a8b0e3bd907732c207c2b534",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.1"
+ },
+ "provide": {
+ "ext-mbstring": "*"
+ },
+ "suggest": {
+ "ext-mbstring": "For best performance"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "1.27-dev"
+ },
+ "thanks": {
+ "name": "symfony/polyfill",
+ "url": "https://github.com/symfony/polyfill"
+ }
+ },
+ "autoload": {
+ "files": [
+ "bootstrap.php"
+ ],
+ "psr-4": {
+ "Symfony\\Polyfill\\Mbstring\\": ""
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony polyfill for the Mbstring extension",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "compatibility",
+ "mbstring",
+ "polyfill",
+ "portable",
+ "shim"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.27.0"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2022-11-03T14:55:06+00:00"
+ },
+ {
+ "name": "symfony/property-access",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/property-access.git",
+ "reference": "5a389172011e2c37b47c896d0b156549126690a1"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/property-access/zipball/5a389172011e2c37b47c896d0b156549126690a1",
+ "reference": "5a389172011e2c37b47c896d0b156549126690a1",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/property-info": "^5.4|^6.0"
+ },
+ "require-dev": {
+ "symfony/cache": "^5.4|^6.0"
+ },
+ "suggest": {
+ "psr/cache-implementation": "To cache access methods."
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\PropertyAccess\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides functions to read and write from/to an object or array using a simple string notation",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "access",
+ "array",
+ "extraction",
+ "index",
+ "injection",
+ "object",
+ "property",
+ "property path",
+ "reflection"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/property-access/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/property-info",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/property-info.git",
+ "reference": "5cf906918ea0f74032ffc5c0b85def246ce409df"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/property-info/zipball/5cf906918ea0f74032ffc5c0b85def246ce409df",
+ "reference": "5cf906918ea0f74032ffc5c0b85def246ce409df",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/string": "^5.4|^6.0"
+ },
+ "conflict": {
+ "phpdocumentor/reflection-docblock": "<5.2",
+ "phpdocumentor/type-resolver": "<1.5.1",
+ "symfony/dependency-injection": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.10.4|^2",
+ "phpdocumentor/reflection-docblock": "^5.2",
+ "phpstan/phpdoc-parser": "^1.0",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/serializer": "^5.4|^6.0"
+ },
+ "suggest": {
+ "phpdocumentor/reflection-docblock": "To use the PHPDoc",
+ "psr/cache-implementation": "To cache results",
+ "symfony/doctrine-bridge": "To use Doctrine metadata",
+ "symfony/serializer": "To use Serializer metadata"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\PropertyInfo\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Kévin Dunglas",
+ "email": "dunglas@gmail.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Extracts information about PHP class' properties using metadata of popular sources",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "doctrine",
+ "phpdoc",
+ "property",
+ "symfony",
+ "type",
+ "validator"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/property-info/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:53:37+00:00"
+ },
+ {
+ "name": "symfony/routing",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/routing.git",
+ "reference": "fa643fa4c56de161f8bc8c0492a76a60140b50e4"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/routing/zipball/fa643fa4c56de161f8bc8c0492a76a60140b50e4",
+ "reference": "fa643fa4c56de161f8bc8c0492a76a60140b50e4",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "conflict": {
+ "doctrine/annotations": "<1.12",
+ "symfony/config": "<6.2",
+ "symfony/dependency-injection": "<5.4",
+ "symfony/yaml": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.12|^2",
+ "psr/log": "^1|^2|^3",
+ "symfony/config": "^6.2",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/yaml": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/config": "For using the all-in-one router or any loader",
+ "symfony/expression-language": "For using expression matching",
+ "symfony/http-foundation": "For using a Symfony Request object",
+ "symfony/yaml": "For using the YAML loader"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Routing\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Maps an HTTP request to a set of configuration variables",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "router",
+ "routing",
+ "uri",
+ "url"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/routing/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:53:37+00:00"
+ },
+ {
+ "name": "symfony/runtime",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/runtime.git",
+ "reference": "111b9d617d0cfc71d44baf01eb9951517fd8b739"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/runtime/zipball/111b9d617d0cfc71d44baf01eb9951517fd8b739",
+ "reference": "111b9d617d0cfc71d44baf01eb9951517fd8b739",
+ "shasum": ""
+ },
+ "require": {
+ "composer-plugin-api": "^1.0|^2.0",
+ "php": ">=8.1"
+ },
+ "conflict": {
+ "symfony/dotenv": "<5.4"
+ },
+ "require-dev": {
+ "composer/composer": "^1.0.2|^2.0",
+ "symfony/console": "^5.4|^6.0",
+ "symfony/dotenv": "^5.4|^6.0",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/http-kernel": "^5.4|^6.0"
+ },
+ "type": "composer-plugin",
+ "extra": {
+ "class": "Symfony\\Component\\Runtime\\Internal\\ComposerPlugin"
+ },
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Runtime\\": "",
+ "Symfony\\Runtime\\Symfony\\Component\\": "Internal/"
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Enables decoupling PHP applications from global state",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/runtime/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-02T07:44:01+00:00"
+ },
+ {
+ "name": "symfony/security-bundle",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/security-bundle.git",
+ "reference": "601bcc14b6e8c168dc5985d31cfdfd11bd07b50b"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/security-bundle/zipball/601bcc14b6e8c168dc5985d31cfdfd11bd07b50b",
+ "reference": "601bcc14b6e8c168dc5985d31cfdfd11bd07b50b",
+ "shasum": ""
+ },
+ "require": {
+ "composer-runtime-api": ">=2.1",
+ "ext-xml": "*",
+ "php": ">=8.1",
+ "symfony/config": "^6.1",
+ "symfony/dependency-injection": "^6.2",
+ "symfony/event-dispatcher": "^5.4|^6.0",
+ "symfony/http-foundation": "^6.2",
+ "symfony/http-kernel": "^6.2",
+ "symfony/password-hasher": "^5.4|^6.0",
+ "symfony/security-core": "^6.2",
+ "symfony/security-csrf": "^5.4|^6.0",
+ "symfony/security-http": "^6.2.6"
+ },
+ "conflict": {
+ "symfony/browser-kit": "<5.4",
+ "symfony/console": "<5.4",
+ "symfony/framework-bundle": "<5.4",
+ "symfony/ldap": "<5.4",
+ "symfony/twig-bundle": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.10.4|^2",
+ "symfony/asset": "^5.4|^6.0",
+ "symfony/browser-kit": "^5.4|^6.0",
+ "symfony/console": "^5.4|^6.0",
+ "symfony/css-selector": "^5.4|^6.0",
+ "symfony/dom-crawler": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/form": "^5.4|^6.0",
+ "symfony/framework-bundle": "^5.4|^6.0",
+ "symfony/ldap": "^5.4|^6.0",
+ "symfony/process": "^5.4|^6.0",
+ "symfony/rate-limiter": "^5.4|^6.0",
+ "symfony/serializer": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/twig-bridge": "^5.4|^6.0",
+ "symfony/twig-bundle": "^5.4|^6.0",
+ "symfony/validator": "^5.4|^6.0",
+ "symfony/yaml": "^5.4|^6.0",
+ "twig/twig": "^2.13|^3.0.4"
+ },
+ "type": "symfony-bundle",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Bundle\\SecurityBundle\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides a tight integration of the Security component into the Symfony full-stack framework",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/security-bundle/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-21T12:32:47+00:00"
+ },
+ {
+ "name": "symfony/security-core",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/security-core.git",
+ "reference": "5dd5509ec58bf30c98811681870f58e7f5918bbe"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/security-core/zipball/5dd5509ec58bf30c98811681870f58e7f5918bbe",
+ "reference": "5dd5509ec58bf30c98811681870f58e7f5918bbe",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/event-dispatcher-contracts": "^1.1|^2|^3",
+ "symfony/password-hasher": "^5.4|^6.0",
+ "symfony/service-contracts": "^1.1.6|^2|^3"
+ },
+ "conflict": {
+ "symfony/event-dispatcher": "<5.4",
+ "symfony/http-foundation": "<5.4",
+ "symfony/ldap": "<5.4",
+ "symfony/security-guard": "<5.4",
+ "symfony/validator": "<5.4"
+ },
+ "require-dev": {
+ "psr/cache": "^1.0|^2.0|^3.0",
+ "psr/container": "^1.1|^2.0",
+ "psr/log": "^1|^2|^3",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/event-dispatcher": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/ldap": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/validator": "^5.4|^6.0"
+ },
+ "suggest": {
+ "psr/container-implementation": "To instantiate the Security class",
+ "symfony/event-dispatcher": "",
+ "symfony/expression-language": "For using the expression voter",
+ "symfony/http-foundation": "",
+ "symfony/ldap": "For using LDAP integration",
+ "symfony/validator": "For using the user password constraint"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Security\\Core\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony Security Component - Core Library",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/security-core/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-17T11:05:34+00:00"
+ },
+ {
+ "name": "symfony/security-csrf",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/security-csrf.git",
+ "reference": "6cce7efdce68e0670d2f19acebc21dcd0798e333"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/security-csrf/zipball/6cce7efdce68e0670d2f19acebc21dcd0798e333",
+ "reference": "6cce7efdce68e0670d2f19acebc21dcd0798e333",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/security-core": "^5.4|^6.0"
+ },
+ "conflict": {
+ "symfony/http-foundation": "<5.4"
+ },
+ "require-dev": {
+ "symfony/http-foundation": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/http-foundation": "For using the class SessionTokenStorage."
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Security\\Csrf\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony Security Component - CSRF Library",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/security-csrf/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-16T09:57:23+00:00"
+ },
+ {
+ "name": "symfony/security-http",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/security-http.git",
+ "reference": "0b96e76243877b53e9ff1418f9e538ecf480dd69"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/security-http/zipball/0b96e76243877b53e9ff1418f9e538ecf480dd69",
+ "reference": "0b96e76243877b53e9ff1418f9e538ecf480dd69",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/http-kernel": "^6.2",
+ "symfony/polyfill-mbstring": "~1.0",
+ "symfony/property-access": "^5.4|^6.0",
+ "symfony/security-core": "~6.0.19|~6.1.11|^6.2.5"
+ },
+ "conflict": {
+ "symfony/event-dispatcher": "<5.4.9|>=6,<6.0.9",
+ "symfony/security-bundle": "<5.4",
+ "symfony/security-csrf": "<5.4"
+ },
+ "require-dev": {
+ "psr/log": "^1|^2|^3",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/rate-limiter": "^5.4|^6.0",
+ "symfony/routing": "^5.4|^6.0",
+ "symfony/security-csrf": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/routing": "For using the HttpUtils class to create sub-requests, redirect the user, and match URLs",
+ "symfony/security-csrf": "For using tokens to protect authentication/logout attempts"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Security\\Http\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Symfony Security Component - HTTP Integration",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/security-http/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-28T10:56:03+00:00"
+ },
+ {
+ "name": "symfony/service-contracts",
+ "version": "v3.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/service-contracts.git",
+ "reference": "a8c9cedf55f314f3a186041d19537303766df09a"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/service-contracts/zipball/a8c9cedf55f314f3a186041d19537303766df09a",
+ "reference": "a8c9cedf55f314f3a186041d19537303766df09a",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "psr/container": "^2.0"
+ },
+ "conflict": {
+ "ext-psr": "<1.1|>=2"
+ },
+ "suggest": {
+ "symfony/service-implementation": ""
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.3-dev"
+ },
+ "thanks": {
+ "name": "symfony/contracts",
+ "url": "https://github.com/symfony/contracts"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Contracts\\Service\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Test/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Generic abstractions related to writing services",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "abstractions",
+ "contracts",
+ "decoupling",
+ "interfaces",
+ "interoperability",
+ "standards"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/service-contracts/tree/v3.2.1"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-01T10:32:47+00:00"
+ },
+ {
+ "name": "symfony/stopwatch",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/stopwatch.git",
+ "reference": "f3adc98c1061875dd2edcd45e5b04e63d0e29f8f"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/stopwatch/zipball/f3adc98c1061875dd2edcd45e5b04e63d0e29f8f",
+ "reference": "f3adc98c1061875dd2edcd45e5b04e63d0e29f8f",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/service-contracts": "^1|^2|^3"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Stopwatch\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides a way to profile code",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/stopwatch/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/string",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/string.git",
+ "reference": "67b8c1eec78296b85dc1c7d9743830160218993d"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/string/zipball/67b8c1eec78296b85dc1c7d9743830160218993d",
+ "reference": "67b8c1eec78296b85dc1c7d9743830160218993d",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/polyfill-ctype": "~1.8",
+ "symfony/polyfill-intl-grapheme": "~1.0",
+ "symfony/polyfill-intl-normalizer": "~1.0",
+ "symfony/polyfill-mbstring": "~1.0"
+ },
+ "conflict": {
+ "symfony/translation-contracts": "<2.0"
+ },
+ "require-dev": {
+ "symfony/error-handler": "^5.4|^6.0",
+ "symfony/http-client": "^5.4|^6.0",
+ "symfony/intl": "^6.2",
+ "symfony/translation-contracts": "^2.0|^3.0",
+ "symfony/var-exporter": "^5.4|^6.0"
+ },
+ "type": "library",
+ "autoload": {
+ "files": [
+ "Resources/functions.php"
+ ],
+ "psr-4": {
+ "Symfony\\Component\\String\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides an object-oriented API to strings and deals with bytes, UTF-8 code points and grapheme clusters in a unified way",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "grapheme",
+ "i18n",
+ "string",
+ "unicode",
+ "utf-8",
+ "utf8"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/string/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-24T10:42:00+00:00"
+ },
+ {
+ "name": "symfony/translation-contracts",
+ "version": "v3.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/translation-contracts.git",
+ "reference": "dfec258b9dd17a6b24420d464c43bffe347441c8"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/translation-contracts/zipball/dfec258b9dd17a6b24420d464c43bffe347441c8",
+ "reference": "dfec258b9dd17a6b24420d464c43bffe347441c8",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "suggest": {
+ "symfony/translation-implementation": ""
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.3-dev"
+ },
+ "thanks": {
+ "name": "symfony/contracts",
+ "url": "https://github.com/symfony/contracts"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Contracts\\Translation\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Test/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Generic abstractions related to translation",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "abstractions",
+ "contracts",
+ "decoupling",
+ "interfaces",
+ "interoperability",
+ "standards"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/translation-contracts/tree/v3.2.1"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-01T10:32:47+00:00"
+ },
+ {
+ "name": "symfony/validator",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/validator.git",
+ "reference": "4b3bd0a9545bdf7ebc84f0a494c05219010bb403"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/validator/zipball/4b3bd0a9545bdf7ebc84f0a494c05219010bb403",
+ "reference": "4b3bd0a9545bdf7ebc84f0a494c05219010bb403",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/deprecation-contracts": "^2.1|^3",
+ "symfony/polyfill-ctype": "~1.8",
+ "symfony/polyfill-mbstring": "~1.0",
+ "symfony/translation-contracts": "^1.1|^2|^3"
+ },
+ "conflict": {
+ "doctrine/annotations": "<1.13",
+ "doctrine/lexer": "<1.1",
+ "phpunit/phpunit": "<5.4.3",
+ "symfony/dependency-injection": "<5.4",
+ "symfony/expression-language": "<5.4",
+ "symfony/http-kernel": "<5.4",
+ "symfony/intl": "<5.4",
+ "symfony/property-info": "<5.4",
+ "symfony/translation": "<5.4",
+ "symfony/yaml": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.13|^2",
+ "egulias/email-validator": "^2.1.10|^3|^4",
+ "symfony/cache": "^5.4|^6.0",
+ "symfony/config": "^5.4|^6.0",
+ "symfony/console": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/finder": "^5.4|^6.0",
+ "symfony/http-client": "^5.4|^6.0",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/http-kernel": "^5.4|^6.0",
+ "symfony/intl": "^5.4|^6.0",
+ "symfony/mime": "^5.4|^6.0",
+ "symfony/property-access": "^5.4|^6.0",
+ "symfony/property-info": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/yaml": "^5.4|^6.0"
+ },
+ "suggest": {
+ "egulias/email-validator": "Strict (RFC compliant) email validation",
+ "psr/cache-implementation": "For using the mapping cache.",
+ "symfony/config": "",
+ "symfony/expression-language": "For using the Expression validator and the ExpressionLanguageSyntax constraints",
+ "symfony/http-foundation": "",
+ "symfony/intl": "",
+ "symfony/property-access": "For accessing properties within comparison constraints",
+ "symfony/property-info": "To automatically add NotNull and Type constraints",
+ "symfony/translation": "For translating validation errors.",
+ "symfony/yaml": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Validator\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides tools to validate values",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/validator/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-24T10:42:00+00:00"
+ },
+ {
+ "name": "symfony/var-dumper",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/var-dumper.git",
+ "reference": "cf8d4ca1ddc1e3cc242375deb8fc23e54f5e2a1e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/var-dumper/zipball/cf8d4ca1ddc1e3cc242375deb8fc23e54f5e2a1e",
+ "reference": "cf8d4ca1ddc1e3cc242375deb8fc23e54f5e2a1e",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/polyfill-mbstring": "~1.0"
+ },
+ "conflict": {
+ "phpunit/phpunit": "<5.4.3",
+ "symfony/console": "<5.4"
+ },
+ "require-dev": {
+ "ext-iconv": "*",
+ "symfony/console": "^5.4|^6.0",
+ "symfony/process": "^5.4|^6.0",
+ "symfony/uid": "^5.4|^6.0",
+ "twig/twig": "^2.13|^3.0.4"
+ },
+ "suggest": {
+ "ext-iconv": "To convert non-UTF-8 strings to UTF-8 (or symfony/polyfill-iconv in case ext-iconv cannot be used).",
+ "ext-intl": "To show region name in time zone dump",
+ "symfony/console": "To use the ServerDumpCommand and/or the bin/var-dump-server script"
+ },
+ "bin": [
+ "Resources/bin/var-dump-server"
+ ],
+ "type": "library",
+ "autoload": {
+ "files": [
+ "Resources/functions/dump.php"
+ ],
+ "psr-4": {
+ "Symfony\\Component\\VarDumper\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides mechanisms for walking through any arbitrary PHP variable",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "debug",
+ "dump"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/var-dumper/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-24T10:42:00+00:00"
+ },
+ {
+ "name": "symfony/var-exporter",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/var-exporter.git",
+ "reference": "86062dd0103530e151588c8f60f5b85a139f1442"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/var-exporter/zipball/86062dd0103530e151588c8f60f5b85a139f1442",
+ "reference": "86062dd0103530e151588c8f60f5b85a139f1442",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "symfony/var-dumper": "^5.4|^6.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\VarExporter\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Allows exporting any serializable PHP data structure to plain PHP code",
+ "homepage": "https://symfony.com",
+ "keywords": [
+ "clone",
+ "construct",
+ "export",
+ "hydrate",
+ "instantiate",
+ "lazy loading",
+ "proxy",
+ "serialize"
+ ],
+ "support": {
+ "source": "https://github.com/symfony/var-exporter/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-24T10:42:00+00:00"
+ },
+ {
+ "name": "symfony/yaml",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/yaml.git",
+ "reference": "e8e6a1d59e050525f27a1f530aa9703423cb7f57"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/yaml/zipball/e8e6a1d59e050525f27a1f530aa9703423cb7f57",
+ "reference": "e8e6a1d59e050525f27a1f530aa9703423cb7f57",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/polyfill-ctype": "^1.8"
+ },
+ "conflict": {
+ "symfony/console": "<5.4"
+ },
+ "require-dev": {
+ "symfony/console": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/console": "For validating YAML files using the lint command"
+ },
+ "bin": [
+ "Resources/bin/yaml-lint"
+ ],
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\Yaml\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Loads and dumps YAML files",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/yaml/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-16T09:57:23+00:00"
+ }
+ ],
+ "packages-dev": [
+ {
+ "name": "myclabs/deep-copy",
+ "version": "1.11.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/myclabs/DeepCopy.git",
+ "reference": "7284c22080590fb39f2ffa3e9057f10a4ddd0e0c"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/myclabs/DeepCopy/zipball/7284c22080590fb39f2ffa3e9057f10a4ddd0e0c",
+ "reference": "7284c22080590fb39f2ffa3e9057f10a4ddd0e0c",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^7.1 || ^8.0"
+ },
+ "conflict": {
+ "doctrine/collections": "<1.6.8",
+ "doctrine/common": "<2.13.3 || >=3,<3.2.2"
+ },
+ "require-dev": {
+ "doctrine/collections": "^1.6.8",
+ "doctrine/common": "^2.13.3 || ^3.2.2",
+ "phpunit/phpunit": "^7.5.20 || ^8.5.23 || ^9.5.13"
+ },
+ "type": "library",
+ "autoload": {
+ "files": [
+ "src/DeepCopy/deep_copy.php"
+ ],
+ "psr-4": {
+ "DeepCopy\\": "src/DeepCopy/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "description": "Create deep copies (clones) of your objects",
+ "keywords": [
+ "clone",
+ "copy",
+ "duplicate",
+ "object",
+ "object graph"
+ ],
+ "support": {
+ "issues": "https://github.com/myclabs/DeepCopy/issues",
+ "source": "https://github.com/myclabs/DeepCopy/tree/1.11.1"
+ },
+ "funding": [
+ {
+ "url": "https://tidelift.com/funding/github/packagist/myclabs/deep-copy",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-08T13:26:56+00:00"
+ },
+ {
+ "name": "nikic/php-parser",
+ "version": "v4.15.4",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/nikic/PHP-Parser.git",
+ "reference": "6bb5176bc4af8bcb7d926f88718db9b96a2d4290"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/6bb5176bc4af8bcb7d926f88718db9b96a2d4290",
+ "reference": "6bb5176bc4af8bcb7d926f88718db9b96a2d4290",
+ "shasum": ""
+ },
+ "require": {
+ "ext-tokenizer": "*",
+ "php": ">=7.0"
+ },
+ "require-dev": {
+ "ircmaxell/php-yacc": "^0.0.7",
+ "phpunit/phpunit": "^6.5 || ^7.0 || ^8.0 || ^9.0"
+ },
+ "bin": [
+ "bin/php-parse"
+ ],
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "4.9-dev"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "PhpParser\\": "lib/PhpParser"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Nikita Popov"
+ }
+ ],
+ "description": "A PHP parser written in PHP",
+ "keywords": [
+ "parser",
+ "php"
+ ],
+ "support": {
+ "issues": "https://github.com/nikic/PHP-Parser/issues",
+ "source": "https://github.com/nikic/PHP-Parser/tree/v4.15.4"
+ },
+ "time": "2023-03-05T19:49:14+00:00"
+ },
+ {
+ "name": "phar-io/manifest",
+ "version": "2.0.3",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/phar-io/manifest.git",
+ "reference": "97803eca37d319dfa7826cc2437fc020857acb53"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/phar-io/manifest/zipball/97803eca37d319dfa7826cc2437fc020857acb53",
+ "reference": "97803eca37d319dfa7826cc2437fc020857acb53",
+ "shasum": ""
+ },
+ "require": {
+ "ext-dom": "*",
+ "ext-phar": "*",
+ "ext-xmlwriter": "*",
+ "phar-io/version": "^3.0.1",
+ "php": "^7.2 || ^8.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "2.0.x-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Arne Blankerts",
+ "email": "arne@blankerts.de",
+ "role": "Developer"
+ },
+ {
+ "name": "Sebastian Heuer",
+ "email": "sebastian@phpeople.de",
+ "role": "Developer"
+ },
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "Developer"
+ }
+ ],
+ "description": "Component for reading phar.io manifest information from a PHP Archive (PHAR)",
+ "support": {
+ "issues": "https://github.com/phar-io/manifest/issues",
+ "source": "https://github.com/phar-io/manifest/tree/2.0.3"
+ },
+ "time": "2021-07-20T11:28:43+00:00"
+ },
+ {
+ "name": "phar-io/version",
+ "version": "3.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/phar-io/version.git",
+ "reference": "4f7fd7836c6f332bb2933569e566a0d6c4cbed74"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/phar-io/version/zipball/4f7fd7836c6f332bb2933569e566a0d6c4cbed74",
+ "reference": "4f7fd7836c6f332bb2933569e566a0d6c4cbed74",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^7.2 || ^8.0"
+ },
+ "type": "library",
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Arne Blankerts",
+ "email": "arne@blankerts.de",
+ "role": "Developer"
+ },
+ {
+ "name": "Sebastian Heuer",
+ "email": "sebastian@phpeople.de",
+ "role": "Developer"
+ },
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "Developer"
+ }
+ ],
+ "description": "Library for handling version information and constraints",
+ "support": {
+ "issues": "https://github.com/phar-io/version/issues",
+ "source": "https://github.com/phar-io/version/tree/3.2.1"
+ },
+ "time": "2022-02-21T01:04:05+00:00"
+ },
+ {
+ "name": "phpstan/phpstan",
+ "version": "1.10.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/phpstan/phpstan.git",
+ "reference": "b10ceb526d9607903c5b2673f1fc8775dbe48975"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/phpstan/phpstan/zipball/b10ceb526d9607903c5b2673f1fc8775dbe48975",
+ "reference": "b10ceb526d9607903c5b2673f1fc8775dbe48975",
+ "shasum": ""
+ },
+ "require": {
+ "php": "^7.2|^8.0"
+ },
+ "conflict": {
+ "phpstan/phpstan-shim": "*"
+ },
+ "bin": [
+ "phpstan",
+ "phpstan.phar"
+ ],
+ "type": "library",
+ "autoload": {
+ "files": [
+ "bootstrap.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "description": "PHPStan - PHP Static Analysis Tool",
+ "keywords": [
+ "dev",
+ "static analysis"
+ ],
+ "support": {
+ "docs": "https://phpstan.org/user-guide/getting-started",
+ "forum": "https://github.com/phpstan/phpstan/discussions",
+ "issues": "https://github.com/phpstan/phpstan/issues",
+ "security": "https://github.com/phpstan/phpstan/security/policy",
+ "source": "https://github.com/phpstan/phpstan-src"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/ondrejmirtes",
+ "type": "github"
+ },
+ {
+ "url": "https://github.com/phpstan",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/phpstan/phpstan",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-16T15:24:20+00:00"
+ },
+ {
+ "name": "phpunit/php-code-coverage",
+ "version": "10.0.2",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/php-code-coverage.git",
+ "reference": "20800e84296ea4732f9a125e08ce86b4004ae3e4"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/php-code-coverage/zipball/20800e84296ea4732f9a125e08ce86b4004ae3e4",
+ "reference": "20800e84296ea4732f9a125e08ce86b4004ae3e4",
+ "shasum": ""
+ },
+ "require": {
+ "ext-dom": "*",
+ "ext-libxml": "*",
+ "ext-xmlwriter": "*",
+ "nikic/php-parser": "^4.15",
+ "php": ">=8.1",
+ "phpunit/php-file-iterator": "^4.0",
+ "phpunit/php-text-template": "^3.0",
+ "sebastian/code-unit-reverse-lookup": "^3.0",
+ "sebastian/complexity": "^3.0",
+ "sebastian/environment": "^6.0",
+ "sebastian/lines-of-code": "^2.0",
+ "sebastian/version": "^4.0",
+ "theseer/tokenizer": "^1.2.0"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "suggest": {
+ "ext-pcov": "PHP extension that provides line coverage",
+ "ext-xdebug": "PHP extension that provides line coverage as well as branch and path coverage"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "10.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Library that provides collection, processing, and rendering functionality for PHP code coverage information.",
+ "homepage": "https://github.com/sebastianbergmann/php-code-coverage",
+ "keywords": [
+ "coverage",
+ "testing",
+ "xunit"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/php-code-coverage/issues",
+ "source": "https://github.com/sebastianbergmann/php-code-coverage/tree/10.0.2"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-03-06T13:00:19+00:00"
+ },
+ {
+ "name": "phpunit/php-file-iterator",
+ "version": "4.0.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/php-file-iterator.git",
+ "reference": "fd9329ab3368f59fe1fe808a189c51086bd4b6bd"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/php-file-iterator/zipball/fd9329ab3368f59fe1fe808a189c51086bd4b6bd",
+ "reference": "fd9329ab3368f59fe1fe808a189c51086bd4b6bd",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "4.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "FilterIterator implementation that filters files based on a list of suffixes.",
+ "homepage": "https://github.com/sebastianbergmann/php-file-iterator/",
+ "keywords": [
+ "filesystem",
+ "iterator"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/php-file-iterator/issues",
+ "source": "https://github.com/sebastianbergmann/php-file-iterator/tree/4.0.1"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-10T16:53:14+00:00"
+ },
+ {
+ "name": "phpunit/php-invoker",
+ "version": "4.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/php-invoker.git",
+ "reference": "f5e568ba02fa5ba0ddd0f618391d5a9ea50b06d7"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/php-invoker/zipball/f5e568ba02fa5ba0ddd0f618391d5a9ea50b06d7",
+ "reference": "f5e568ba02fa5ba0ddd0f618391d5a9ea50b06d7",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "ext-pcntl": "*",
+ "phpunit/phpunit": "^10.0"
+ },
+ "suggest": {
+ "ext-pcntl": "*"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "4.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Invoke callables with a timeout",
+ "homepage": "https://github.com/sebastianbergmann/php-invoker/",
+ "keywords": [
+ "process"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/php-invoker/issues",
+ "source": "https://github.com/sebastianbergmann/php-invoker/tree/4.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T06:56:09+00:00"
+ },
+ {
+ "name": "phpunit/php-text-template",
+ "version": "3.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/php-text-template.git",
+ "reference": "9f3d3709577a527025f55bcf0f7ab8052c8bb37d"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/php-text-template/zipball/9f3d3709577a527025f55bcf0f7ab8052c8bb37d",
+ "reference": "9f3d3709577a527025f55bcf0f7ab8052c8bb37d",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Simple template engine.",
+ "homepage": "https://github.com/sebastianbergmann/php-text-template/",
+ "keywords": [
+ "template"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/php-text-template/issues",
+ "source": "https://github.com/sebastianbergmann/php-text-template/tree/3.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T06:56:46+00:00"
+ },
+ {
+ "name": "phpunit/php-timer",
+ "version": "6.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/php-timer.git",
+ "reference": "e2a2d67966e740530f4a3343fe2e030ffdc1161d"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/php-timer/zipball/e2a2d67966e740530f4a3343fe2e030ffdc1161d",
+ "reference": "e2a2d67966e740530f4a3343fe2e030ffdc1161d",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "6.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Utility class for timing",
+ "homepage": "https://github.com/sebastianbergmann/php-timer/",
+ "keywords": [
+ "timer"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/php-timer/issues",
+ "source": "https://github.com/sebastianbergmann/php-timer/tree/6.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T06:57:52+00:00"
+ },
+ {
+ "name": "phpunit/phpunit",
+ "version": "10.0.18",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/phpunit.git",
+ "reference": "582563ed2edc62d1455cdbe00ea49fe09428eef3"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/582563ed2edc62d1455cdbe00ea49fe09428eef3",
+ "reference": "582563ed2edc62d1455cdbe00ea49fe09428eef3",
+ "shasum": ""
+ },
+ "require": {
+ "ext-dom": "*",
+ "ext-json": "*",
+ "ext-libxml": "*",
+ "ext-mbstring": "*",
+ "ext-xml": "*",
+ "ext-xmlwriter": "*",
+ "myclabs/deep-copy": "^1.10.1",
+ "phar-io/manifest": "^2.0.3",
+ "phar-io/version": "^3.0.2",
+ "php": ">=8.1",
+ "phpunit/php-code-coverage": "^10.0",
+ "phpunit/php-file-iterator": "^4.0",
+ "phpunit/php-invoker": "^4.0",
+ "phpunit/php-text-template": "^3.0",
+ "phpunit/php-timer": "^6.0",
+ "sebastian/cli-parser": "^2.0",
+ "sebastian/code-unit": "^2.0",
+ "sebastian/comparator": "^5.0",
+ "sebastian/diff": "^5.0",
+ "sebastian/environment": "^6.0",
+ "sebastian/exporter": "^5.0",
+ "sebastian/global-state": "^6.0",
+ "sebastian/object-enumerator": "^5.0",
+ "sebastian/recursion-context": "^5.0",
+ "sebastian/type": "^4.0",
+ "sebastian/version": "^4.0"
+ },
+ "suggest": {
+ "ext-soap": "To be able to generate mocks based on WSDL files"
+ },
+ "bin": [
+ "phpunit"
+ ],
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "10.0-dev"
+ }
+ },
+ "autoload": {
+ "files": [
+ "src/Framework/Assert/Functions.php"
+ ],
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "The PHP Unit Testing framework.",
+ "homepage": "https://phpunit.de/",
+ "keywords": [
+ "phpunit",
+ "testing",
+ "xunit"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/phpunit/issues",
+ "security": "https://github.com/sebastianbergmann/phpunit/security/policy",
+ "source": "https://github.com/sebastianbergmann/phpunit/tree/10.0.18"
+ },
+ "funding": [
+ {
+ "url": "https://phpunit.de/sponsors.html",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/phpunit/phpunit",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-03-22T06:15:31+00:00"
+ },
+ {
+ "name": "sebastian/cli-parser",
+ "version": "2.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/cli-parser.git",
+ "reference": "efdc130dbbbb8ef0b545a994fd811725c5282cae"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/cli-parser/zipball/efdc130dbbbb8ef0b545a994fd811725c5282cae",
+ "reference": "efdc130dbbbb8ef0b545a994fd811725c5282cae",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "2.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Library for parsing CLI options",
+ "homepage": "https://github.com/sebastianbergmann/cli-parser",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/cli-parser/issues",
+ "source": "https://github.com/sebastianbergmann/cli-parser/tree/2.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T06:58:15+00:00"
+ },
+ {
+ "name": "sebastian/code-unit",
+ "version": "2.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/code-unit.git",
+ "reference": "a81fee9eef0b7a76af11d121767abc44c104e503"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/code-unit/zipball/a81fee9eef0b7a76af11d121767abc44c104e503",
+ "reference": "a81fee9eef0b7a76af11d121767abc44c104e503",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "2.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Collection of value objects that represent the PHP code units",
+ "homepage": "https://github.com/sebastianbergmann/code-unit",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/code-unit/issues",
+ "source": "https://github.com/sebastianbergmann/code-unit/tree/2.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T06:58:43+00:00"
+ },
+ {
+ "name": "sebastian/code-unit-reverse-lookup",
+ "version": "3.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/code-unit-reverse-lookup.git",
+ "reference": "5e3a687f7d8ae33fb362c5c0743794bbb2420a1d"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/code-unit-reverse-lookup/zipball/5e3a687f7d8ae33fb362c5c0743794bbb2420a1d",
+ "reference": "5e3a687f7d8ae33fb362c5c0743794bbb2420a1d",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ }
+ ],
+ "description": "Looks up which function or method a line of code belongs to",
+ "homepage": "https://github.com/sebastianbergmann/code-unit-reverse-lookup/",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/code-unit-reverse-lookup/issues",
+ "source": "https://github.com/sebastianbergmann/code-unit-reverse-lookup/tree/3.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T06:59:15+00:00"
+ },
+ {
+ "name": "sebastian/comparator",
+ "version": "5.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/comparator.git",
+ "reference": "72f01e6586e0caf6af81297897bd112eb7e9627c"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/comparator/zipball/72f01e6586e0caf6af81297897bd112eb7e9627c",
+ "reference": "72f01e6586e0caf6af81297897bd112eb7e9627c",
+ "shasum": ""
+ },
+ "require": {
+ "ext-dom": "*",
+ "ext-mbstring": "*",
+ "php": ">=8.1",
+ "sebastian/diff": "^5.0",
+ "sebastian/exporter": "^5.0"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "5.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ },
+ {
+ "name": "Jeff Welch",
+ "email": "whatthejeff@gmail.com"
+ },
+ {
+ "name": "Volker Dusch",
+ "email": "github@wallbash.com"
+ },
+ {
+ "name": "Bernhard Schussek",
+ "email": "bschussek@2bepublished.at"
+ }
+ ],
+ "description": "Provides the functionality to compare PHP values for equality",
+ "homepage": "https://github.com/sebastianbergmann/comparator",
+ "keywords": [
+ "comparator",
+ "compare",
+ "equality"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/comparator/issues",
+ "source": "https://github.com/sebastianbergmann/comparator/tree/5.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:07:16+00:00"
+ },
+ {
+ "name": "sebastian/complexity",
+ "version": "3.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/complexity.git",
+ "reference": "e67d240970c9dc7ea7b2123a6d520e334dd61dc6"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/complexity/zipball/e67d240970c9dc7ea7b2123a6d520e334dd61dc6",
+ "reference": "e67d240970c9dc7ea7b2123a6d520e334dd61dc6",
+ "shasum": ""
+ },
+ "require": {
+ "nikic/php-parser": "^4.10",
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Library for calculating the complexity of PHP code units",
+ "homepage": "https://github.com/sebastianbergmann/complexity",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/complexity/issues",
+ "source": "https://github.com/sebastianbergmann/complexity/tree/3.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T06:59:47+00:00"
+ },
+ {
+ "name": "sebastian/diff",
+ "version": "5.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/diff.git",
+ "reference": "70dd1b20bc198da394ad542e988381b44e64e39f"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/diff/zipball/70dd1b20bc198da394ad542e988381b44e64e39f",
+ "reference": "70dd1b20bc198da394ad542e988381b44e64e39f",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0",
+ "symfony/process": "^4.2 || ^5"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "5.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ },
+ {
+ "name": "Kore Nordmann",
+ "email": "mail@kore-nordmann.de"
+ }
+ ],
+ "description": "Diff implementation",
+ "homepage": "https://github.com/sebastianbergmann/diff",
+ "keywords": [
+ "diff",
+ "udiff",
+ "unidiff",
+ "unified diff"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/diff/issues",
+ "source": "https://github.com/sebastianbergmann/diff/tree/5.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:00:31+00:00"
+ },
+ {
+ "name": "sebastian/environment",
+ "version": "6.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/environment.git",
+ "reference": "b6f3694c6386c7959915a0037652e0c40f6f69cc"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/environment/zipball/b6f3694c6386c7959915a0037652e0c40f6f69cc",
+ "reference": "b6f3694c6386c7959915a0037652e0c40f6f69cc",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "suggest": {
+ "ext-posix": "*"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "6.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ }
+ ],
+ "description": "Provides functionality to handle HHVM/PHP environments",
+ "homepage": "https://github.com/sebastianbergmann/environment",
+ "keywords": [
+ "Xdebug",
+ "environment",
+ "hhvm"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/environment/issues",
+ "source": "https://github.com/sebastianbergmann/environment/tree/6.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:03:04+00:00"
+ },
+ {
+ "name": "sebastian/exporter",
+ "version": "5.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/exporter.git",
+ "reference": "f3ec4bf931c0b31e5b413f5b4fc970a7d03338c0"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/exporter/zipball/f3ec4bf931c0b31e5b413f5b4fc970a7d03338c0",
+ "reference": "f3ec4bf931c0b31e5b413f5b4fc970a7d03338c0",
+ "shasum": ""
+ },
+ "require": {
+ "ext-mbstring": "*",
+ "php": ">=8.1",
+ "sebastian/recursion-context": "^5.0"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "5.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ },
+ {
+ "name": "Jeff Welch",
+ "email": "whatthejeff@gmail.com"
+ },
+ {
+ "name": "Volker Dusch",
+ "email": "github@wallbash.com"
+ },
+ {
+ "name": "Adam Harvey",
+ "email": "aharvey@php.net"
+ },
+ {
+ "name": "Bernhard Schussek",
+ "email": "bschussek@gmail.com"
+ }
+ ],
+ "description": "Provides the functionality to export PHP variables for visualization",
+ "homepage": "https://www.github.com/sebastianbergmann/exporter",
+ "keywords": [
+ "export",
+ "exporter"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/exporter/issues",
+ "source": "https://github.com/sebastianbergmann/exporter/tree/5.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:06:49+00:00"
+ },
+ {
+ "name": "sebastian/global-state",
+ "version": "6.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/global-state.git",
+ "reference": "aab257c712de87b90194febd52e4d184551c2d44"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/global-state/zipball/aab257c712de87b90194febd52e4d184551c2d44",
+ "reference": "aab257c712de87b90194febd52e4d184551c2d44",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "sebastian/object-reflector": "^3.0",
+ "sebastian/recursion-context": "^5.0"
+ },
+ "require-dev": {
+ "ext-dom": "*",
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "6.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ }
+ ],
+ "description": "Snapshotting of global state",
+ "homepage": "http://www.github.com/sebastianbergmann/global-state",
+ "keywords": [
+ "global state"
+ ],
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/global-state/issues",
+ "source": "https://github.com/sebastianbergmann/global-state/tree/6.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:07:38+00:00"
+ },
+ {
+ "name": "sebastian/lines-of-code",
+ "version": "2.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/lines-of-code.git",
+ "reference": "17c4d940ecafb3d15d2cf916f4108f664e28b130"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/lines-of-code/zipball/17c4d940ecafb3d15d2cf916f4108f664e28b130",
+ "reference": "17c4d940ecafb3d15d2cf916f4108f664e28b130",
+ "shasum": ""
+ },
+ "require": {
+ "nikic/php-parser": "^4.10",
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "2.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Library for counting the lines of code in PHP source code",
+ "homepage": "https://github.com/sebastianbergmann/lines-of-code",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/lines-of-code/issues",
+ "source": "https://github.com/sebastianbergmann/lines-of-code/tree/2.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:08:02+00:00"
+ },
+ {
+ "name": "sebastian/object-enumerator",
+ "version": "5.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/object-enumerator.git",
+ "reference": "202d0e344a580d7f7d04b3fafce6933e59dae906"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/object-enumerator/zipball/202d0e344a580d7f7d04b3fafce6933e59dae906",
+ "reference": "202d0e344a580d7f7d04b3fafce6933e59dae906",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "sebastian/object-reflector": "^3.0",
+ "sebastian/recursion-context": "^5.0"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "5.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ }
+ ],
+ "description": "Traverses array structures and object graphs to enumerate all referenced objects",
+ "homepage": "https://github.com/sebastianbergmann/object-enumerator/",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/object-enumerator/issues",
+ "source": "https://github.com/sebastianbergmann/object-enumerator/tree/5.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:08:32+00:00"
+ },
+ {
+ "name": "sebastian/object-reflector",
+ "version": "3.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/object-reflector.git",
+ "reference": "24ed13d98130f0e7122df55d06c5c4942a577957"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/object-reflector/zipball/24ed13d98130f0e7122df55d06c5c4942a577957",
+ "reference": "24ed13d98130f0e7122df55d06c5c4942a577957",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "3.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ }
+ ],
+ "description": "Allows reflection of object attributes, including inherited and non-public ones",
+ "homepage": "https://github.com/sebastianbergmann/object-reflector/",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/object-reflector/issues",
+ "source": "https://github.com/sebastianbergmann/object-reflector/tree/3.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:06:18+00:00"
+ },
+ {
+ "name": "sebastian/recursion-context",
+ "version": "5.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/recursion-context.git",
+ "reference": "05909fb5bc7df4c52992396d0116aed689f93712"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/recursion-context/zipball/05909fb5bc7df4c52992396d0116aed689f93712",
+ "reference": "05909fb5bc7df4c52992396d0116aed689f93712",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "5.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de"
+ },
+ {
+ "name": "Jeff Welch",
+ "email": "whatthejeff@gmail.com"
+ },
+ {
+ "name": "Adam Harvey",
+ "email": "aharvey@php.net"
+ }
+ ],
+ "description": "Provides functionality to recursively process PHP variables",
+ "homepage": "https://github.com/sebastianbergmann/recursion-context",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/recursion-context/issues",
+ "source": "https://github.com/sebastianbergmann/recursion-context/tree/5.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:05:40+00:00"
+ },
+ {
+ "name": "sebastian/type",
+ "version": "4.0.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/type.git",
+ "reference": "462699a16464c3944eefc02ebdd77882bd3925bf"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/type/zipball/462699a16464c3944eefc02ebdd77882bd3925bf",
+ "reference": "462699a16464c3944eefc02ebdd77882bd3925bf",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^10.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "4.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Collection of value objects that represent the types of the PHP type system",
+ "homepage": "https://github.com/sebastianbergmann/type",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/type/issues",
+ "source": "https://github.com/sebastianbergmann/type/tree/4.0.0"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-03T07:10:45+00:00"
+ },
+ {
+ "name": "sebastian/version",
+ "version": "4.0.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/sebastianbergmann/version.git",
+ "reference": "c51fa83a5d8f43f1402e3f32a005e6262244ef17"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/sebastianbergmann/version/zipball/c51fa83a5d8f43f1402e3f32a005e6262244ef17",
+ "reference": "c51fa83a5d8f43f1402e3f32a005e6262244ef17",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-main": "4.0-dev"
+ }
+ },
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Sebastian Bergmann",
+ "email": "sebastian@phpunit.de",
+ "role": "lead"
+ }
+ ],
+ "description": "Library that helps with managing the version number of Git-hosted PHP projects",
+ "homepage": "https://github.com/sebastianbergmann/version",
+ "support": {
+ "issues": "https://github.com/sebastianbergmann/version/issues",
+ "source": "https://github.com/sebastianbergmann/version/tree/4.0.1"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/sebastianbergmann",
+ "type": "github"
+ }
+ ],
+ "time": "2023-02-07T11:34:05+00:00"
+ },
+ {
+ "name": "squizlabs/php_codesniffer",
+ "version": "3.7.2",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/squizlabs/PHP_CodeSniffer.git",
+ "reference": "ed8e00df0a83aa96acf703f8c2979ff33341f879"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/squizlabs/PHP_CodeSniffer/zipball/ed8e00df0a83aa96acf703f8c2979ff33341f879",
+ "reference": "ed8e00df0a83aa96acf703f8c2979ff33341f879",
+ "shasum": ""
+ },
+ "require": {
+ "ext-simplexml": "*",
+ "ext-tokenizer": "*",
+ "ext-xmlwriter": "*",
+ "php": ">=5.4.0"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^4.0 || ^5.0 || ^6.0 || ^7.0"
+ },
+ "bin": [
+ "bin/phpcs",
+ "bin/phpcbf"
+ ],
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "3.x-dev"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Greg Sherwood",
+ "role": "lead"
+ }
+ ],
+ "description": "PHP_CodeSniffer tokenizes PHP, JavaScript and CSS files and detects violations of a defined set of coding standards.",
+ "homepage": "https://github.com/squizlabs/PHP_CodeSniffer",
+ "keywords": [
+ "phpcs",
+ "standards",
+ "static analysis"
+ ],
+ "support": {
+ "issues": "https://github.com/squizlabs/PHP_CodeSniffer/issues",
+ "source": "https://github.com/squizlabs/PHP_CodeSniffer",
+ "wiki": "https://github.com/squizlabs/PHP_CodeSniffer/wiki"
+ },
+ "time": "2023-02-22T23:07:41+00:00"
+ },
+ {
+ "name": "symfony/browser-kit",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/browser-kit.git",
+ "reference": "87bd43240e6cc855f70ea1c7a448ab3bd442633c"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/browser-kit/zipball/87bd43240e6cc855f70ea1c7a448ab3bd442633c",
+ "reference": "87bd43240e6cc855f70ea1c7a448ab3bd442633c",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/dom-crawler": "^5.4|^6.0"
+ },
+ "require-dev": {
+ "symfony/css-selector": "^5.4|^6.0",
+ "symfony/http-client": "^5.4|^6.0",
+ "symfony/mime": "^5.4|^6.0",
+ "symfony/process": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/process": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\BrowserKit\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Simulates the behavior of a web browser, allowing you to make requests, click on links and submit forms programmatically",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/browser-kit/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/dom-crawler",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/dom-crawler.git",
+ "reference": "65a906f5141ff2d3aef1b01c128fba78f5e9f921"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/dom-crawler/zipball/65a906f5141ff2d3aef1b01c128fba78f5e9f921",
+ "reference": "65a906f5141ff2d3aef1b01c128fba78f5e9f921",
+ "shasum": ""
+ },
+ "require": {
+ "masterminds/html5": "^2.6",
+ "php": ">=8.1",
+ "symfony/polyfill-ctype": "~1.8",
+ "symfony/polyfill-mbstring": "~1.0"
+ },
+ "require-dev": {
+ "symfony/css-selector": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/css-selector": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Component\\DomCrawler\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Eases DOM navigation for HTML and XML documents",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/dom-crawler/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/phpunit-bridge",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/phpunit-bridge.git",
+ "reference": "56965fae0b6b8d271015990eff5240ffff02e185"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/phpunit-bridge/zipball/56965fae0b6b8d271015990eff5240ffff02e185",
+ "reference": "56965fae0b6b8d271015990eff5240ffff02e185",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.1.3"
+ },
+ "conflict": {
+ "phpunit/phpunit": "<7.5|9.1.2"
+ },
+ "require-dev": {
+ "symfony/deprecation-contracts": "^2.1|^3.0",
+ "symfony/error-handler": "^5.4|^6.0"
+ },
+ "suggest": {
+ "symfony/error-handler": "For tracking deprecated interfaces usages at runtime with DebugClassLoader"
+ },
+ "bin": [
+ "bin/simple-phpunit"
+ ],
+ "type": "symfony-bridge",
+ "extra": {
+ "thanks": {
+ "name": "phpunit/phpunit",
+ "url": "https://github.com/sebastianbergmann/phpunit"
+ }
+ },
+ "autoload": {
+ "files": [
+ "bootstrap.php"
+ ],
+ "psr-4": {
+ "Symfony\\Bridge\\PhpUnit\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Nicolas Grekas",
+ "email": "p@tchwork.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides utilities for PHPUnit, especially user deprecation notices management",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/phpunit-bridge/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-16T09:57:23+00:00"
+ },
+ {
+ "name": "symfony/twig-bridge",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/twig-bridge.git",
+ "reference": "f1899fd3b8a29f9544440a716a1ed7d1121c4615"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/twig-bridge/zipball/f1899fd3b8a29f9544440a716a1ed7d1121c4615",
+ "reference": "f1899fd3b8a29f9544440a716a1ed7d1121c4615",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/translation-contracts": "^1.1|^2|^3",
+ "twig/twig": "^2.13|^3.0.4"
+ },
+ "conflict": {
+ "phpdocumentor/reflection-docblock": "<3.2.2",
+ "phpdocumentor/type-resolver": "<1.4.0",
+ "symfony/console": "<5.4",
+ "symfony/form": "<6.2.7",
+ "symfony/http-foundation": "<5.4",
+ "symfony/http-kernel": "<6.2",
+ "symfony/mime": "<6.2",
+ "symfony/translation": "<5.4",
+ "symfony/workflow": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.12|^2",
+ "egulias/email-validator": "^2.1.10|^3|^4",
+ "league/html-to-markdown": "^5.0",
+ "phpdocumentor/reflection-docblock": "^3.0|^4.0|^5.0",
+ "symfony/asset": "^5.4|^6.0",
+ "symfony/console": "^5.4|^6.0",
+ "symfony/dependency-injection": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/finder": "^5.4|^6.0",
+ "symfony/form": "^6.2.7",
+ "symfony/html-sanitizer": "^6.1",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/http-kernel": "^6.2",
+ "symfony/intl": "^5.4|^6.0",
+ "symfony/mime": "^6.2",
+ "symfony/polyfill-intl-icu": "~1.0",
+ "symfony/property-info": "^5.4|^6.0",
+ "symfony/routing": "^5.4|^6.0",
+ "symfony/security-acl": "^2.8|^3.0",
+ "symfony/security-core": "^5.4|^6.0",
+ "symfony/security-csrf": "^5.4|^6.0",
+ "symfony/security-http": "^5.4|^6.0",
+ "symfony/serializer": "^6.2",
+ "symfony/stopwatch": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/web-link": "^5.4|^6.0",
+ "symfony/workflow": "^5.4|^6.0",
+ "symfony/yaml": "^5.4|^6.0",
+ "twig/cssinliner-extra": "^2.12|^3",
+ "twig/inky-extra": "^2.12|^3",
+ "twig/markdown-extra": "^2.12|^3"
+ },
+ "suggest": {
+ "symfony/asset": "For using the AssetExtension",
+ "symfony/expression-language": "For using the ExpressionExtension",
+ "symfony/finder": "",
+ "symfony/form": "For using the FormExtension",
+ "symfony/html-sanitizer": "For using the HtmlSanitizerExtension",
+ "symfony/http-kernel": "For using the HttpKernelExtension",
+ "symfony/routing": "For using the RoutingExtension",
+ "symfony/security-core": "For using the SecurityExtension",
+ "symfony/security-csrf": "For using the CsrfExtension",
+ "symfony/security-http": "For using the LogoutUrlExtension",
+ "symfony/stopwatch": "For using the StopwatchExtension",
+ "symfony/translation": "For using the TranslationExtension",
+ "symfony/var-dumper": "For using the DumpExtension",
+ "symfony/web-link": "For using the WebLinkExtension",
+ "symfony/yaml": "For using the YamlExtension"
+ },
+ "type": "symfony-bridge",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Bridge\\Twig\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides integration for Twig with various Symfony components",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/twig-bridge/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-24T10:42:00+00:00"
+ },
+ {
+ "name": "symfony/twig-bundle",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/twig-bundle.git",
+ "reference": "8bb562655c6ae4b8fae9cf72077591f38b961566"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/twig-bundle/zipball/8bb562655c6ae4b8fae9cf72077591f38b961566",
+ "reference": "8bb562655c6ae4b8fae9cf72077591f38b961566",
+ "shasum": ""
+ },
+ "require": {
+ "composer-runtime-api": ">=2.1",
+ "php": ">=8.1",
+ "symfony/config": "^6.1",
+ "symfony/dependency-injection": "^6.1",
+ "symfony/http-foundation": "^5.4|^6.0",
+ "symfony/http-kernel": "^6.2",
+ "symfony/twig-bridge": "^6.2",
+ "twig/twig": "^2.13|^3.0.4"
+ },
+ "conflict": {
+ "symfony/framework-bundle": "<5.4",
+ "symfony/translation": "<5.4"
+ },
+ "require-dev": {
+ "doctrine/annotations": "^1.10.4|^2",
+ "symfony/asset": "^5.4|^6.0",
+ "symfony/expression-language": "^5.4|^6.0",
+ "symfony/finder": "^5.4|^6.0",
+ "symfony/form": "^5.4|^6.0",
+ "symfony/framework-bundle": "^5.4|^6.0",
+ "symfony/routing": "^5.4|^6.0",
+ "symfony/stopwatch": "^5.4|^6.0",
+ "symfony/translation": "^5.4|^6.0",
+ "symfony/web-link": "^5.4|^6.0",
+ "symfony/yaml": "^5.4|^6.0"
+ },
+ "type": "symfony-bundle",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Bundle\\TwigBundle\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides a tight integration of Twig into the Symfony full-stack framework",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/twig-bundle/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-14T08:44:56+00:00"
+ },
+ {
+ "name": "symfony/web-profiler-bundle",
+ "version": "v6.2.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/symfony/web-profiler-bundle.git",
+ "reference": "0d183e0a69652e348007e97ffff8d3ded9cc6d2d"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/symfony/web-profiler-bundle/zipball/0d183e0a69652e348007e97ffff8d3ded9cc6d2d",
+ "reference": "0d183e0a69652e348007e97ffff8d3ded9cc6d2d",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=8.1",
+ "symfony/config": "^5.4|^6.0",
+ "symfony/framework-bundle": "^5.4|^6.0",
+ "symfony/http-kernel": "^6.1",
+ "symfony/routing": "^5.4|^6.0",
+ "symfony/twig-bundle": "^5.4|^6.0",
+ "twig/twig": "^2.13|^3.0.4"
+ },
+ "conflict": {
+ "symfony/form": "<5.4",
+ "symfony/mailer": "<5.4",
+ "symfony/messenger": "<5.4"
+ },
+ "require-dev": {
+ "symfony/browser-kit": "^5.4|^6.0",
+ "symfony/console": "^5.4|^6.0",
+ "symfony/css-selector": "^5.4|^6.0",
+ "symfony/stopwatch": "^5.4|^6.0"
+ },
+ "type": "symfony-bundle",
+ "autoload": {
+ "psr-4": {
+ "Symfony\\Bundle\\WebProfilerBundle\\": ""
+ },
+ "exclude-from-classmap": [
+ "/Tests/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ },
+ {
+ "name": "Symfony Community",
+ "homepage": "https://symfony.com/contributors"
+ }
+ ],
+ "description": "Provides a development tool that gives detailed information about the execution of any request",
+ "homepage": "https://symfony.com",
+ "support": {
+ "source": "https://github.com/symfony/web-profiler-bundle/tree/v6.2.7"
+ },
+ "funding": [
+ {
+ "url": "https://symfony.com/sponsor",
+ "type": "custom"
+ },
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-21T16:32:03+00:00"
+ },
+ {
+ "name": "theseer/tokenizer",
+ "version": "1.2.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/theseer/tokenizer.git",
+ "reference": "34a41e998c2183e22995f158c581e7b5e755ab9e"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/theseer/tokenizer/zipball/34a41e998c2183e22995f158c581e7b5e755ab9e",
+ "reference": "34a41e998c2183e22995f158c581e7b5e755ab9e",
+ "shasum": ""
+ },
+ "require": {
+ "ext-dom": "*",
+ "ext-tokenizer": "*",
+ "ext-xmlwriter": "*",
+ "php": "^7.2 || ^8.0"
+ },
+ "type": "library",
+ "autoload": {
+ "classmap": [
+ "src/"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Arne Blankerts",
+ "email": "arne@blankerts.de",
+ "role": "Developer"
+ }
+ ],
+ "description": "A small library for converting tokenized PHP source code into XML and potentially other formats",
+ "support": {
+ "issues": "https://github.com/theseer/tokenizer/issues",
+ "source": "https://github.com/theseer/tokenizer/tree/1.2.1"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/theseer",
+ "type": "github"
+ }
+ ],
+ "time": "2021-07-28T10:34:58+00:00"
+ },
+ {
+ "name": "twig/twig",
+ "version": "v3.5.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/twigphp/Twig.git",
+ "reference": "a6e0510cc793912b451fd40ab983a1d28f611c15"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/twigphp/Twig/zipball/a6e0510cc793912b451fd40ab983a1d28f611c15",
+ "reference": "a6e0510cc793912b451fd40ab983a1d28f611c15",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=7.2.5",
+ "symfony/polyfill-ctype": "^1.8",
+ "symfony/polyfill-mbstring": "^1.3"
+ },
+ "require-dev": {
+ "psr/container": "^1.0",
+ "symfony/phpunit-bridge": "^4.4.9|^5.0.9|^6.0"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "3.5-dev"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "Twig\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "BSD-3-Clause"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com",
+ "homepage": "http://fabien.potencier.org",
+ "role": "Lead Developer"
+ },
+ {
+ "name": "Twig Team",
+ "role": "Contributors"
+ },
+ {
+ "name": "Armin Ronacher",
+ "email": "armin.ronacher@active-4.com",
+ "role": "Project Founder"
+ }
+ ],
+ "description": "Twig, the flexible, fast, and secure template language for PHP",
+ "homepage": "https://twig.symfony.com",
+ "keywords": [
+ "templating"
+ ],
+ "support": {
+ "issues": "https://github.com/twigphp/Twig/issues",
+ "source": "https://github.com/twigphp/Twig/tree/v3.5.1"
+ },
+ "funding": [
+ {
+ "url": "https://github.com/fabpot",
+ "type": "github"
+ },
+ {
+ "url": "https://tidelift.com/funding/github/packagist/twig/twig",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2023-02-08T07:49:20+00:00"
+ }
+ ],
+ "aliases": [],
+ "minimum-stability": "dev",
+ "stability-flags": [],
+ "prefer-stable": true,
+ "prefer-lowest": false,
+ "platform": {
+ "php": "^8.2",
+ "ext-ctype": "*",
+ "ext-iconv": "*"
+ },
+ "platform-dev": [],
+ "plugin-api-version": "2.3.0"
+}
diff --git a/config/bundles.php b/config/bundles.php
new file mode 100644
index 0000000..5c0e9a9
--- /dev/null
+++ b/config/bundles.php
@@ -0,0 +1,11 @@
+ ['all' => true],
+ Doctrine\Bundle\DoctrineBundle\DoctrineBundle::class => ['all' => true],
+ Doctrine\Bundle\MigrationsBundle\DoctrineMigrationsBundle::class => ['all' => true],
+ Symfony\Bundle\TwigBundle\TwigBundle::class => ['all' => true],
+ Symfony\Bundle\WebProfilerBundle\WebProfilerBundle::class => ['dev' => true, 'test' => true],
+ Symfony\Bundle\SecurityBundle\SecurityBundle::class => ['all' => true],
+ Ser\DTORequestBundle\DTORequestBundle::class => ['all' => true],
+];
diff --git a/config/mapping/Company.orm.yml b/config/mapping/Company.orm.yml
new file mode 100644
index 0000000..065f8a5
--- /dev/null
+++ b/config/mapping/Company.orm.yml
@@ -0,0 +1,5 @@
+RashinMe\Entity\Company:
+ type: embeddable
+ fields:
+ name: { type: string }
+ url: { type: string }
diff --git a/config/mapping/Education.orm.yml b/config/mapping/Education.orm.yml
new file mode 100644
index 0000000..9792c58
--- /dev/null
+++ b/config/mapping/Education.orm.yml
@@ -0,0 +1,25 @@
+RashinMe\Entity\Education:
+ type: entity
+ table: educations
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ institution:
+ type: string
+ length: 175
+ faculty:
+ type: string
+ length: 175
+ specialization:
+ type: string
+ length: 175
+ fromDate:
+ type: datetime
+ toDate:
+ type: datetime
+ nullable: true
diff --git a/config/mapping/File.orm.yml b/config/mapping/File.orm.yml
new file mode 100644
index 0000000..3a5373d
--- /dev/null
+++ b/config/mapping/File.orm.yml
@@ -0,0 +1,25 @@
+RashinMe\Entity\File:
+ type: entity
+ table: files
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ name:
+ type: string
+ length: 200
+ path:
+ type: string
+ length: 200
+ mimeType:
+ type: string
+ length: 25
+ size:
+ type: integer
+ options:
+ default: 0
+ unsigned: true
diff --git a/config/mapping/Job.orm.yml b/config/mapping/Job.orm.yml
new file mode 100644
index 0000000..2feb359
--- /dev/null
+++ b/config/mapping/Job.orm.yml
@@ -0,0 +1,27 @@
+RashinMe\Entity\Job:
+ type: entity
+ table: jobs
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ name:
+ type: string
+ length: 175
+ type:
+ type: string
+ length: 30
+ description:
+ type: text
+ fromDate:
+ type: datetime
+ toDate:
+ type: datetime
+ nullable: true
+ embedded:
+ company:
+ class: RashinMe\Entity\Company
diff --git a/config/mapping/Link.orm.yml b/config/mapping/Link.orm.yml
new file mode 100644
index 0000000..26ce6d9
--- /dev/null
+++ b/config/mapping/Link.orm.yml
@@ -0,0 +1,27 @@
+RashinMe\Entity\Link:
+ type: entity
+ table: links
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ title:
+ type: string
+ length: 175
+ url:
+ type: string
+ length: 175
+ manyToOne:
+ project:
+ targetEntity: RashinMe\Entity\Project
+ inversedBy: links
+ joinColumn:
+ name: project_id
+ referencedColumnName: id
+ nullable: false
+ cascade: [remove]
+ onDelete: CASCADE
diff --git a/config/mapping/Project.orm.yml b/config/mapping/Project.orm.yml
new file mode 100644
index 0000000..daebfb3
--- /dev/null
+++ b/config/mapping/Project.orm.yml
@@ -0,0 +1,46 @@
+RashinMe\Entity\Project:
+ type: entity
+ table: projects
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ name:
+ type: string
+ length: 175
+ description:
+ type: text
+ oneToOne:
+ image:
+ targetEntity: RashinMe\Entity\File
+ joinColumn:
+ name: image_id
+ referencedColumnName: id
+ cascade:
+ - all
+ manyToMany:
+ tags:
+ targetEntity: RashinMe\Entity\Tag
+ joinTable:
+ name: project_tags
+ joinColumns:
+ project_id:
+ referencedColumnName: id
+ inverseJoinColumns:
+ tag_id:
+ referencedColumnName: id
+ cascade:
+ - persist
+ oneToMany:
+ links:
+ targetEntity: RashinMe\Entity\Link
+ mappedBy: project
+ orderBy: { "id": "ASC" }
+ cascade:
+ - persist
+ - merge
+ orphanRemoval: true
diff --git a/config/mapping/Property.orm.yml b/config/mapping/Property.orm.yml
new file mode 100644
index 0000000..183f0dd
--- /dev/null
+++ b/config/mapping/Property.orm.yml
@@ -0,0 +1,13 @@
+RashinMe\Entity\Property:
+ type: entity
+ table: properties
+ id:
+ key:
+ type: string
+ length: 200
+ generator:
+ strategy: NONE
+ fields:
+ value:
+ type: string
+ length: 200
diff --git a/config/mapping/Section.orm.yml b/config/mapping/Section.orm.yml
new file mode 100644
index 0000000..1c4cea0
--- /dev/null
+++ b/config/mapping/Section.orm.yml
@@ -0,0 +1,23 @@
+RashinMe\Entity\Section:
+ type: entity
+ table: sections
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ name:
+ type: string
+ length: 175
+ oneToMany:
+ skills:
+ targetEntity: RashinMe\Entity\Skill
+ mappedBy: section
+ orderBy: { "id": "ASC" }
+ cascade:
+ - persist
+ - remove
+ orphanRemoval: true
diff --git a/config/mapping/Skill.orm.yml b/config/mapping/Skill.orm.yml
new file mode 100644
index 0000000..18bb522
--- /dev/null
+++ b/config/mapping/Skill.orm.yml
@@ -0,0 +1,31 @@
+RashinMe\Entity\Skill:
+ type: entity
+ table: skills
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ name:
+ type: string
+ length: 175
+ description:
+ type: text
+ oneToOne:
+ image:
+ targetEntity: RashinMe\Entity\File
+ joinColumn:
+ name: image_id
+ referencedColumnName: id
+ manyToOne:
+ section:
+ targetEntity: RashinMe\Entity\Section
+ joinColumn:
+ name: section_id
+ referencedColumnName: id
+ nullable: false
+ cascade: ["persist", "merge"]
+ onDelete: CASCADE
diff --git a/config/mapping/Tag.orm.yml b/config/mapping/Tag.orm.yml
new file mode 100644
index 0000000..630a762
--- /dev/null
+++ b/config/mapping/Tag.orm.yml
@@ -0,0 +1,14 @@
+RashinMe\Entity\Tag:
+ type: entity
+ table: tags
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ name:
+ type: string
+ length: 75
diff --git a/config/mapping/User.orm.yml b/config/mapping/User.orm.yml
new file mode 100644
index 0000000..d807077
--- /dev/null
+++ b/config/mapping/User.orm.yml
@@ -0,0 +1,26 @@
+RashinMe\Entity\User:
+ type: entity
+ table: users
+ id:
+ id:
+ type: integer
+ generator:
+ strategy: IDENTITY
+ options:
+ unsigned: true
+ fields:
+ firstName:
+ type: string
+ length: 75
+ lastName:
+ type: string
+ length: 75
+ email:
+ type: string
+ length: 175
+ unique: true
+ password:
+ type: string
+ length: 175
+ roles:
+ type: json
diff --git a/config/packages/cache.yaml b/config/packages/cache.yaml
new file mode 100644
index 0000000..1d1e412
--- /dev/null
+++ b/config/packages/cache.yaml
@@ -0,0 +1,24 @@
+framework:
+ cache:
+ # Unique name of your app: used to compute stable namespaces for cache keys.
+ #prefix_seed: your_vendor_name/app_name
+
+ # The "app" cache stores to the filesystem by default.
+ # The data in this cache should persist between deploys.
+ # Other options include:
+
+ # Redis
+ #app: cache.adapter.redis
+ #default_redis_provider: redis://localhost
+
+ # APCu (not recommended with heavy random-write workloads as memory fragmentation can cause perf issues)
+ #app: cache.adapter.apcu
+
+ # Namespaced pools use the above "app" backend by default
+ #pools:
+ #my.dedicated.cache: null
+
+#when@prod:
+# framework:
+# cache:
+# default_redis_provider: '%env(REDIS_URL)%'
diff --git a/config/packages/doctrine.yaml b/config/packages/doctrine.yaml
new file mode 100644
index 0000000..6949cdf
--- /dev/null
+++ b/config/packages/doctrine.yaml
@@ -0,0 +1,51 @@
+doctrine:
+ dbal:
+ dbname: '%env(resolve:DB_NAME)%'
+ host: '%env(resolve:DB_HOST)%'
+ port: '%env(resolve:DB_PORT)%'
+ user: '%env(resolve:DB_USER)%'
+ password: '%env(resolve:DB_PASSWORD)%'
+ driver: '%env(resolve:DB_DRIVER)%'
+ server_version: '%env(resolve:DB_SERVER_VERSION)%'
+ charset: UTF8
+ orm:
+ auto_generate_proxy_classes: true
+ enable_lazy_ghost_objects: true
+ naming_strategy: doctrine.orm.naming_strategy.underscore_number_aware
+ auto_mapping: false
+ mappings:
+ RashinMe:
+ is_bundle: false
+ type: yml
+ dir: '%kernel.project_dir%/config/mapping/'
+ prefix: 'RashinMe\Entity'
+ alias: RashinMe
+
+when@test:
+ doctrine:
+ dbal:
+ # "TEST_TOKEN" is typically set by ParaTest
+# dbname_suffix: '_test%env(default::TEST_TOKEN)%'
+ connections:
+ default:
+ driver: pdo_sqlite
+ memory: true
+when@prod:
+ doctrine:
+ orm:
+ auto_generate_proxy_classes: false
+ proxy_dir: '%kernel.build_dir%/doctrine/orm/Proxies'
+ query_cache_driver:
+ type: pool
+ pool: doctrine.system_cache_pool
+ result_cache_driver:
+ type: pool
+ pool: doctrine.result_cache_pool
+
+ framework:
+ cache:
+ pools:
+ doctrine.result_cache_pool:
+ adapter: cache.app
+ doctrine.system_cache_pool:
+ adapter: cache.system
diff --git a/config/packages/doctrine_migrations.yaml b/config/packages/doctrine_migrations.yaml
new file mode 100644
index 0000000..29231d9
--- /dev/null
+++ b/config/packages/doctrine_migrations.yaml
@@ -0,0 +1,6 @@
+doctrine_migrations:
+ migrations_paths:
+ # namespace is arbitrary but should be different from App\Migrations
+ # as migrations classes should NOT be autoloaded
+ 'DoctrineMigrations': '%kernel.project_dir%/migrations'
+ enable_profiler: false
diff --git a/config/packages/framework.yaml b/config/packages/framework.yaml
new file mode 100644
index 0000000..6d85c29
--- /dev/null
+++ b/config/packages/framework.yaml
@@ -0,0 +1,25 @@
+# see https://symfony.com/doc/current/reference/configuration/framework.html
+framework:
+ secret: '%env(APP_SECRET)%'
+ #csrf_protection: true
+ http_method_override: false
+ handle_all_throwables: true
+
+ # Enables session support. Note that the session will ONLY be started if you read or write from it.
+ # Remove or comment this section to explicitly disable session support.
+ session:
+ handler_id: null
+ cookie_secure: auto
+ cookie_samesite: lax
+ storage_factory_id: session.storage.factory.native
+
+ #esi: true
+ #fragments: true
+ php_errors:
+ log: true
+
+when@test:
+ framework:
+ test: true
+ session:
+ storage_factory_id: session.storage.factory.mock_file
diff --git a/config/packages/routing.yaml b/config/packages/routing.yaml
new file mode 100644
index 0000000..4b766ce
--- /dev/null
+++ b/config/packages/routing.yaml
@@ -0,0 +1,12 @@
+framework:
+ router:
+ utf8: true
+
+ # Configure how to generate URLs in non-HTTP contexts, such as CLI commands.
+ # See https://symfony.com/doc/current/routing.html#generating-urls-in-commands
+ #default_uri: http://localhost
+
+when@prod:
+ framework:
+ router:
+ strict_requirements: null
diff --git a/config/packages/security.yaml b/config/packages/security.yaml
new file mode 100644
index 0000000..a684549
--- /dev/null
+++ b/config/packages/security.yaml
@@ -0,0 +1,61 @@
+security:
+ # https://symfony.com/doc/current/security.html#registering-the-user-hashing-passwords
+ password_hashers:
+ Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface: 'auto'
+ # https://symfony.com/doc/current/security.html#loading-the-user-the-user-provider
+ providers:
+ app_user_provider:
+ entity:
+ class: RashinMe\Entity\User
+ property: email
+ firewalls:
+ dev:
+ pattern: ^/(_(profiler|wdt)|css|images|js)/
+ security: false
+ main:
+ lazy: true
+ pattern: ^/
+ json_login:
+ login_path: auth_login
+ check_path: auth_login
+ username_path: email
+ password_path: password
+# logout:
+# path: auth_logout
+ provider: app_user_provider
+# custom_authenticators:
+# - RashinMe\Authenticator\LoginAuthenticator
+
+ # activate different ways to authenticate
+ # https://symfony.com/doc/current/security.html#the-firewall
+
+ # https://symfony.com/doc/current/security/impersonating_user.html
+ # switch_user: true
+
+ # Easy way to control access for large sections of your site
+ # Note: Only the *first* access control that matches will be used
+ access_decision_manager:
+ strategy: unanimous
+ allow_if_all_abstain: false
+ access_control:
+# - { path: ^/api/login, roles: PUBLIC_ACCESS }
+# - { path: ^/api, roles: IS_AUTHENTICATED_FULLY }
+# - { path: ^/, roles: PUBLIC_ACCESS }
+
+# - { path: ^/api/login, roles: IS_AUTHENTICATED_ANONYMOUSLY }
+
+ # - { path: ^/admin, roles: ROLE_ADMIN }
+ # - { path: ^/profile, roles: ROLE_USER }
+
+when@test:
+ security:
+ password_hashers:
+ # By default, password hashers are resource intensive and take time. This is
+ # important to generate secure password hashes. In tests however, secure hashes
+ # are not important, waste resources and increase test times. The following
+ # reduces the work factor to the lowest possible values.
+ Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface:
+ algorithm: auto
+ cost: 4 # Lowest possible value for bcrypt
+ time_cost: 3 # Lowest possible value for argon
+ memory_cost: 10 # Lowest possible value for argon
diff --git a/config/packages/twig.yaml b/config/packages/twig.yaml
new file mode 100644
index 0000000..583c79c
--- /dev/null
+++ b/config/packages/twig.yaml
@@ -0,0 +1,6 @@
+twig:
+ paths:
+ '%kernel.project_dir%/src/Resources/Templates': ~
+when@test:
+ twig:
+ strict_variables: true
diff --git a/config/packages/validator.yaml b/config/packages/validator.yaml
new file mode 100644
index 0000000..0201281
--- /dev/null
+++ b/config/packages/validator.yaml
@@ -0,0 +1,13 @@
+framework:
+ validation:
+ email_validation_mode: html5
+
+ # Enables validator auto-mapping support.
+ # For instance, basic validation constraints will be inferred from Doctrine's metadata.
+ #auto_mapping:
+ # App\Entity\: []
+
+when@test:
+ framework:
+ validation:
+ not_compromised_password: false
diff --git a/config/packages/web_profiler.yaml b/config/packages/web_profiler.yaml
new file mode 100644
index 0000000..b946111
--- /dev/null
+++ b/config/packages/web_profiler.yaml
@@ -0,0 +1,17 @@
+when@dev:
+ web_profiler:
+ toolbar: true
+ intercept_redirects: false
+
+ framework:
+ profiler:
+ only_exceptions: false
+ collect_serializer_data: true
+
+when@test:
+ web_profiler:
+ toolbar: false
+ intercept_redirects: false
+
+ framework:
+ profiler: { collect: false }
diff --git a/config/preload.php b/config/preload.php
new file mode 100644
index 0000000..5ebcdb2
--- /dev/null
+++ b/config/preload.php
@@ -0,0 +1,5 @@
+
+
+
+ Sergey Rashin - Portfolio
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/admin/package.json b/frontend/admin/package.json
new file mode 100644
index 0000000..9d2077d
--- /dev/null
+++ b/frontend/admin/package.json
@@ -0,0 +1,44 @@
+{
+ "name": "rashinme-frontend-admin",
+ "version": "0.1.0",
+ "private": true,
+ "dependencies": {
+ "@mui/icons-material": "^5.11.16",
+ "@mui/material": "^5.13.1",
+ "axios": "^1.4.0",
+ "prop-types": "^15.8.1",
+ "query-string": "^8.1.0",
+ "ra-data-simple-rest": "^4.10.3",
+ "ra-input-rich-text": "^4.10.3",
+ "react": "^18.2.0",
+ "react-admin": "^4.10.3",
+ "react-dom": "^18.2.0",
+ "react-router-dom": "^6.11.2",
+ "web-vitals": "^3.3.1"
+ },
+ "devDependencies": {
+ "@testing-library/jest-dom": "^5.16.5",
+ "@testing-library/react": "^14.0.0",
+ "@testing-library/user-event": "^14.4.3",
+ "@types/react": "^18.2.6",
+ "@types/react-dom": "^18.2.4",
+ "@types/react-router-dom": "^5.3.3",
+ "@vitejs/plugin-react": "^4.0.0",
+ "eslint": "^8.41.0",
+ "eslint-config-airbnb": "^19.0.4",
+ "eslint-plugin-import": "^2.27.5",
+ "eslint-plugin-jsx-a11y": "^6.7.1",
+ "eslint-plugin-react": "^7.32.2",
+ "eslint-plugin-react-hooks": "^4.6.0",
+ "react-error-overlay": "^6.0.11",
+ "source-map-explorer": "^2.5.3",
+ "typescript": "^5.0.4",
+ "vite": "^4.3.8"
+ },
+ "scripts": {
+ "analyze": "source-map-explorer 'build/static/js/*.js'",
+ "dev": "vite",
+ "build": "vite build",
+ "preview": "vite preview"
+ }
+}
diff --git a/frontend/admin/pnpm-lock.yaml b/frontend/admin/pnpm-lock.yaml
new file mode 100644
index 0000000..e4e861f
--- /dev/null
+++ b/frontend/admin/pnpm-lock.yaml
@@ -0,0 +1,4491 @@
+lockfileVersion: '6.0'
+
+dependencies:
+ '@mui/icons-material':
+ specifier: ^5.11.16
+ version: 5.11.16(@mui/material@5.13.1)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/material':
+ specifier: ^5.13.1
+ version: 5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0)
+ axios:
+ specifier: ^1.4.0
+ version: 1.4.0
+ prop-types:
+ specifier: ^15.8.1
+ version: 15.8.1
+ query-string:
+ specifier: ^8.1.0
+ version: 8.1.0
+ ra-data-simple-rest:
+ specifier: ^4.10.3
+ version: 4.10.3(ra-core@4.10.3)
+ ra-input-rich-text:
+ specifier: ^4.10.3
+ version: 4.10.3(@mui/icons-material@5.11.16)(@mui/material@5.13.1)(ra-core@4.10.3)(ra-ui-materialui@4.10.3)(react-dom@18.2.0)(react@18.2.0)
+ react:
+ specifier: ^18.2.0
+ version: 18.2.0
+ react-admin:
+ specifier: ^4.10.3
+ version: 4.10.3(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0)
+ react-dom:
+ specifier: ^18.2.0
+ version: 18.2.0(react@18.2.0)
+ react-router-dom:
+ specifier: ^6.11.2
+ version: 6.11.2(react-dom@18.2.0)(react@18.2.0)
+ web-vitals:
+ specifier: ^3.3.1
+ version: 3.3.1
+
+devDependencies:
+ '@testing-library/jest-dom':
+ specifier: ^5.16.5
+ version: 5.16.5
+ '@testing-library/react':
+ specifier: ^14.0.0
+ version: 14.0.0(react-dom@18.2.0)(react@18.2.0)
+ '@testing-library/user-event':
+ specifier: ^14.4.3
+ version: 14.4.3(@testing-library/dom@9.3.0)
+ '@types/react':
+ specifier: ^18.2.6
+ version: 18.2.6
+ '@types/react-dom':
+ specifier: ^18.2.4
+ version: 18.2.4
+ '@types/react-router-dom':
+ specifier: ^5.3.3
+ version: 5.3.3
+ '@vitejs/plugin-react':
+ specifier: ^4.0.0
+ version: 4.0.0(vite@4.3.8)
+ eslint:
+ specifier: ^8.41.0
+ version: 8.41.0
+ eslint-config-airbnb:
+ specifier: ^19.0.4
+ version: 19.0.4(eslint-plugin-import@2.27.5)(eslint-plugin-jsx-a11y@6.7.1)(eslint-plugin-react-hooks@4.6.0)(eslint-plugin-react@7.32.2)(eslint@8.41.0)
+ eslint-plugin-import:
+ specifier: ^2.27.5
+ version: 2.27.5(eslint@8.41.0)
+ eslint-plugin-jsx-a11y:
+ specifier: ^6.7.1
+ version: 6.7.1(eslint@8.41.0)
+ eslint-plugin-react:
+ specifier: ^7.32.2
+ version: 7.32.2(eslint@8.41.0)
+ eslint-plugin-react-hooks:
+ specifier: ^4.6.0
+ version: 4.6.0(eslint@8.41.0)
+ react-error-overlay:
+ specifier: ^6.0.11
+ version: 6.0.11
+ source-map-explorer:
+ specifier: ^2.5.3
+ version: 2.5.3
+ typescript:
+ specifier: ^5.0.4
+ version: 5.0.4
+ vite:
+ specifier: ^4.3.8
+ version: 4.3.8
+
+packages:
+
+ /@adobe/css-tools@4.2.0:
+ resolution: {integrity: sha512-E09FiIft46CmH5Qnjb0wsW54/YQd69LsxeKUOWawmws1XWvyFGURnAChH0mlr7YPFR1ofwvUQfcL0J3lMxXqPA==}
+ dev: true
+
+ /@ampproject/remapping@2.2.1:
+ resolution: {integrity: sha512-lFMjJTrFL3j7L9yBxwYfCq2k6qqwHyzuUl/XBnif78PWTJYyL/dfowQHWE3sp6U6ZzqWiiIZnpTMO96zhkjwtg==}
+ engines: {node: '>=6.0.0'}
+ dependencies:
+ '@jridgewell/gen-mapping': 0.3.3
+ '@jridgewell/trace-mapping': 0.3.18
+
+ /@babel/code-frame@7.21.4:
+ resolution: {integrity: sha512-LYvhNKfwWSPpocw8GI7gpK2nq3HSDuEPC/uSYaALSJu9xjsalaaYFOq0Pwt5KmVqwEbZlDu81aLXwBOmD/Fv9g==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/highlight': 7.18.6
+
+ /@babel/compat-data@7.21.7:
+ resolution: {integrity: sha512-KYMqFYTaenzMK4yUtf4EW9wc4N9ef80FsbMtkwool5zpwl4YrT1SdWYSTRcT94KO4hannogdS+LxY7L+arP3gA==}
+ engines: {node: '>=6.9.0'}
+
+ /@babel/core@7.21.8:
+ resolution: {integrity: sha512-YeM22Sondbo523Sz0+CirSPnbj9bG3P0CdHcBZdqUuaeOaYEFbOLoGU7lebvGP6P5J/WE9wOn7u7C4J9HvS1xQ==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@ampproject/remapping': 2.2.1
+ '@babel/code-frame': 7.21.4
+ '@babel/generator': 7.21.5
+ '@babel/helper-compilation-targets': 7.21.5(@babel/core@7.21.8)
+ '@babel/helper-module-transforms': 7.21.5
+ '@babel/helpers': 7.21.5
+ '@babel/parser': 7.21.8
+ '@babel/template': 7.20.7
+ '@babel/traverse': 7.21.5
+ '@babel/types': 7.21.5
+ convert-source-map: 1.9.0
+ debug: 4.3.4
+ gensync: 1.0.0-beta.2
+ json5: 2.2.3
+ semver: 6.3.0
+ transitivePeerDependencies:
+ - supports-color
+
+ /@babel/generator@7.21.5:
+ resolution: {integrity: sha512-SrKK/sRv8GesIW1bDagf9cCG38IOMYZusoe1dfg0D8aiUe3Amvoj1QtjTPAWcfrZFvIwlleLb0gxzQidL9w14w==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/types': 7.21.5
+ '@jridgewell/gen-mapping': 0.3.3
+ '@jridgewell/trace-mapping': 0.3.18
+ jsesc: 2.5.2
+
+ /@babel/helper-compilation-targets@7.21.5(@babel/core@7.21.8):
+ resolution: {integrity: sha512-1RkbFGUKex4lvsB9yhIfWltJM5cZKUftB2eNajaDv3dCMEp49iBG0K14uH8NnX9IPux2+mK7JGEOB0jn48/J6w==}
+ engines: {node: '>=6.9.0'}
+ peerDependencies:
+ '@babel/core': ^7.0.0
+ dependencies:
+ '@babel/compat-data': 7.21.7
+ '@babel/core': 7.21.8
+ '@babel/helper-validator-option': 7.21.0
+ browserslist: 4.21.5
+ lru-cache: 5.1.1
+ semver: 6.3.0
+
+ /@babel/helper-environment-visitor@7.21.5:
+ resolution: {integrity: sha512-IYl4gZ3ETsWocUWgsFZLM5i1BYx9SoemminVEXadgLBa9TdeorzgLKm8wWLA6J1N/kT3Kch8XIk1laNzYoHKvQ==}
+ engines: {node: '>=6.9.0'}
+
+ /@babel/helper-function-name@7.21.0:
+ resolution: {integrity: sha512-HfK1aMRanKHpxemaY2gqBmL04iAPOPRj7DxtNbiDOrJK+gdwkiNRVpCpUJYbUT+aZyemKN8brqTOxzCaG6ExRg==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/template': 7.20.7
+ '@babel/types': 7.21.5
+
+ /@babel/helper-hoist-variables@7.18.6:
+ resolution: {integrity: sha512-UlJQPkFqFULIcyW5sbzgbkxn2FKRgwWiRexcuaR8RNJRy8+LLveqPjwZV/bwrLZCN0eUHD/x8D0heK1ozuoo6Q==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/types': 7.21.5
+
+ /@babel/helper-module-imports@7.21.4:
+ resolution: {integrity: sha512-orajc5T2PsRYUN3ZryCEFeMDYwyw09c/pZeaQEZPH0MpKzSvn3e0uXsDBu3k03VI+9DBiRo+l22BfKTpKwa/Wg==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/types': 7.21.5
+
+ /@babel/helper-module-transforms@7.21.5:
+ resolution: {integrity: sha512-bI2Z9zBGY2q5yMHoBvJ2a9iX3ZOAzJPm7Q8Yz6YeoUjU/Cvhmi2G4QyTNyPBqqXSgTjUxRg3L0xV45HvkNWWBw==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/helper-environment-visitor': 7.21.5
+ '@babel/helper-module-imports': 7.21.4
+ '@babel/helper-simple-access': 7.21.5
+ '@babel/helper-split-export-declaration': 7.18.6
+ '@babel/helper-validator-identifier': 7.19.1
+ '@babel/template': 7.20.7
+ '@babel/traverse': 7.21.5
+ '@babel/types': 7.21.5
+ transitivePeerDependencies:
+ - supports-color
+
+ /@babel/helper-plugin-utils@7.21.5:
+ resolution: {integrity: sha512-0WDaIlXKOX/3KfBK/dwP1oQGiPh6rjMkT7HIRv7i5RR2VUMwrx5ZL0dwBkKx7+SW1zwNdgjHd34IMk5ZjTeHVg==}
+ engines: {node: '>=6.9.0'}
+
+ /@babel/helper-simple-access@7.21.5:
+ resolution: {integrity: sha512-ENPDAMC1wAjR0uaCUwliBdiSl1KBJAVnMTzXqi64c2MG8MPR6ii4qf7bSXDqSFbr4W6W028/rf5ivoHop5/mkg==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/types': 7.21.5
+
+ /@babel/helper-split-export-declaration@7.18.6:
+ resolution: {integrity: sha512-bde1etTx6ZyTmobl9LLMMQsaizFVZrquTEHOqKeQESMKo4PlObf+8+JA25ZsIpZhT/WEd39+vOdLXAFG/nELpA==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/types': 7.21.5
+
+ /@babel/helper-string-parser@7.21.5:
+ resolution: {integrity: sha512-5pTUx3hAJaZIdW99sJ6ZUUgWq/Y+Hja7TowEnLNMm1VivRgZQL3vpBY3qUACVsvw+yQU6+YgfBVmcbLaZtrA1w==}
+ engines: {node: '>=6.9.0'}
+
+ /@babel/helper-validator-identifier@7.19.1:
+ resolution: {integrity: sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==}
+ engines: {node: '>=6.9.0'}
+
+ /@babel/helper-validator-option@7.21.0:
+ resolution: {integrity: sha512-rmL/B8/f0mKS2baE9ZpyTcTavvEuWhTTW8amjzXNvYG4AwBsqTLikfXsEofsJEfKHf+HQVQbFOHy6o+4cnC/fQ==}
+ engines: {node: '>=6.9.0'}
+
+ /@babel/helpers@7.21.5:
+ resolution: {integrity: sha512-BSY+JSlHxOmGsPTydUkPf1MdMQ3M81x5xGCOVgWM3G8XH77sJ292Y2oqcp0CbbgxhqBuI46iUz1tT7hqP7EfgA==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/template': 7.20.7
+ '@babel/traverse': 7.21.5
+ '@babel/types': 7.21.5
+ transitivePeerDependencies:
+ - supports-color
+
+ /@babel/highlight@7.18.6:
+ resolution: {integrity: sha512-u7stbOuYjaPezCuLj29hNW1v64M2Md2qupEKP1fHc7WdOA3DgLh37suiSrZYY7haUB7iBeQZ9P1uiRF359do3g==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/helper-validator-identifier': 7.19.1
+ chalk: 2.4.2
+ js-tokens: 4.0.0
+
+ /@babel/parser@7.21.8:
+ resolution: {integrity: sha512-6zavDGdzG3gUqAdWvlLFfk+36RilI+Pwyuuh7HItyeScCWP3k6i8vKclAQ0bM/0y/Kz/xiwvxhMv9MgTJP5gmA==}
+ engines: {node: '>=6.0.0'}
+ hasBin: true
+ dependencies:
+ '@babel/types': 7.21.5
+
+ /@babel/plugin-proposal-export-namespace-from@7.18.9(@babel/core@7.21.8):
+ resolution: {integrity: sha512-k1NtHyOMvlDDFeb9G5PhUXuGj8m/wiwojgQVEhJ/fsVsMCpLyOP4h0uGEjYJKrRI+EVPlb5Jk+Gt9P97lOGwtA==}
+ engines: {node: '>=6.9.0'}
+ peerDependencies:
+ '@babel/core': ^7.0.0-0
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/helper-plugin-utils': 7.21.5
+ '@babel/plugin-syntax-export-namespace-from': 7.8.3(@babel/core@7.21.8)
+ dev: false
+
+ /@babel/plugin-syntax-dynamic-import@7.8.3(@babel/core@7.21.8):
+ resolution: {integrity: sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==}
+ peerDependencies:
+ '@babel/core': ^7.0.0-0
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/helper-plugin-utils': 7.21.5
+ dev: false
+
+ /@babel/plugin-syntax-export-namespace-from@7.8.3(@babel/core@7.21.8):
+ resolution: {integrity: sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==}
+ peerDependencies:
+ '@babel/core': ^7.0.0-0
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/helper-plugin-utils': 7.21.5
+ dev: false
+
+ /@babel/plugin-transform-modules-commonjs@7.21.5(@babel/core@7.21.8):
+ resolution: {integrity: sha512-OVryBEgKUbtqMoB7eG2rs6UFexJi6Zj6FDXx+esBLPTCxCNxAY9o+8Di7IsUGJ+AVhp5ncK0fxWUBd0/1gPhrQ==}
+ engines: {node: '>=6.9.0'}
+ peerDependencies:
+ '@babel/core': ^7.0.0-0
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/helper-module-transforms': 7.21.5
+ '@babel/helper-plugin-utils': 7.21.5
+ '@babel/helper-simple-access': 7.21.5
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /@babel/plugin-transform-react-jsx-self@7.21.0(@babel/core@7.21.8):
+ resolution: {integrity: sha512-f/Eq+79JEu+KUANFks9UZCcvydOOGMgF7jBrcwjHa5jTZD8JivnhCJYvmlhR/WTXBWonDExPoW0eO/CR4QJirA==}
+ engines: {node: '>=6.9.0'}
+ peerDependencies:
+ '@babel/core': ^7.0.0-0
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/helper-plugin-utils': 7.21.5
+ dev: true
+
+ /@babel/plugin-transform-react-jsx-source@7.19.6(@babel/core@7.21.8):
+ resolution: {integrity: sha512-RpAi004QyMNisst/pvSanoRdJ4q+jMCWyk9zdw/CyLB9j8RXEahodR6l2GyttDRyEVWZtbN+TpLiHJ3t34LbsQ==}
+ engines: {node: '>=6.9.0'}
+ peerDependencies:
+ '@babel/core': ^7.0.0-0
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/helper-plugin-utils': 7.21.5
+ dev: true
+
+ /@babel/runtime@7.21.5:
+ resolution: {integrity: sha512-8jI69toZqqcsnqGGqwGS4Qb1VwLOEp4hz+CXPywcvjs60u3B4Pom/U/7rm4W8tMOYEB+E9wgD0mW1l3r8qlI9Q==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ regenerator-runtime: 0.13.11
+
+ /@babel/template@7.20.7:
+ resolution: {integrity: sha512-8SegXApWe6VoNw0r9JHpSteLKTpTiLZ4rMlGIm9JQ18KiCtyQiAMEazujAHrUS5flrcqYZa75ukev3P6QmUwUw==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/code-frame': 7.21.4
+ '@babel/parser': 7.21.8
+ '@babel/types': 7.21.5
+
+ /@babel/traverse@7.21.5:
+ resolution: {integrity: sha512-AhQoI3YjWi6u/y/ntv7k48mcrCXmus0t79J9qPNlk/lAsFlCiJ047RmbfMOawySTHtywXhbXgpx/8nXMYd+oFw==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/code-frame': 7.21.4
+ '@babel/generator': 7.21.5
+ '@babel/helper-environment-visitor': 7.21.5
+ '@babel/helper-function-name': 7.21.0
+ '@babel/helper-hoist-variables': 7.18.6
+ '@babel/helper-split-export-declaration': 7.18.6
+ '@babel/parser': 7.21.8
+ '@babel/types': 7.21.5
+ debug: 4.3.4
+ globals: 11.12.0
+ transitivePeerDependencies:
+ - supports-color
+
+ /@babel/types@7.21.5:
+ resolution: {integrity: sha512-m4AfNvVF2mVC/F7fDEdH2El3HzUg9It/XsCxZiOTTA3m3qYfcSVSbTfM6Q9xG+hYDniZssYhlXKKUMD5m8tF4Q==}
+ engines: {node: '>=6.9.0'}
+ dependencies:
+ '@babel/helper-string-parser': 7.21.5
+ '@babel/helper-validator-identifier': 7.19.1
+ to-fast-properties: 2.0.0
+
+ /@emotion/babel-plugin@11.11.0:
+ resolution: {integrity: sha512-m4HEDZleaaCH+XgDDsPF15Ht6wTLsgDTeR3WYj9Q/k76JtWhrJjcP4+/XlG8LGT/Rol9qUfOIztXeA84ATpqPQ==}
+ dependencies:
+ '@babel/helper-module-imports': 7.21.4
+ '@babel/runtime': 7.21.5
+ '@emotion/hash': 0.9.1
+ '@emotion/memoize': 0.8.1
+ '@emotion/serialize': 1.1.2
+ babel-plugin-macros: 3.1.0
+ convert-source-map: 1.9.0
+ escape-string-regexp: 4.0.0
+ find-root: 1.1.0
+ source-map: 0.5.7
+ stylis: 4.2.0
+ dev: false
+
+ /@emotion/cache@11.11.0:
+ resolution: {integrity: sha512-P34z9ssTCBi3e9EI1ZsWpNHcfY1r09ZO0rZbRO2ob3ZQMnFI35jB536qoXbkdesr5EUhYi22anuEJuyxifaqAQ==}
+ dependencies:
+ '@emotion/memoize': 0.8.1
+ '@emotion/sheet': 1.2.2
+ '@emotion/utils': 1.2.1
+ '@emotion/weak-memoize': 0.3.1
+ stylis: 4.2.0
+ dev: false
+
+ /@emotion/hash@0.9.1:
+ resolution: {integrity: sha512-gJB6HLm5rYwSLI6PQa+X1t5CFGrv1J1TWG+sOyMCeKz2ojaj6Fnl/rZEspogG+cvqbt4AE/2eIyD2QfLKTBNlQ==}
+ dev: false
+
+ /@emotion/is-prop-valid@1.2.1:
+ resolution: {integrity: sha512-61Mf7Ufx4aDxx1xlDeOm8aFFigGHE4z+0sKCa+IHCeZKiyP9RLD0Mmx7m8b9/Cf37f7NAvQOOJAbQQGVr5uERw==}
+ dependencies:
+ '@emotion/memoize': 0.8.1
+ dev: false
+
+ /@emotion/memoize@0.8.1:
+ resolution: {integrity: sha512-W2P2c/VRW1/1tLox0mVUalvnWXxavmv/Oum2aPsRcoDJuob75FC3Y8FbpfLwUegRcxINtGUMPq0tFCvYNTBXNA==}
+ dev: false
+
+ /@emotion/react@11.11.0(@types/react@18.2.6)(react@18.2.0):
+ resolution: {integrity: sha512-ZSK3ZJsNkwfjT3JpDAWJZlrGD81Z3ytNDsxw1LKq1o+xkmO5pnWfr6gmCC8gHEFf3nSSX/09YrG67jybNPxSUw==}
+ peerDependencies:
+ '@types/react': '*'
+ react: '>=16.8.0'
+ peerDependenciesMeta:
+ '@types/react':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@emotion/babel-plugin': 11.11.0
+ '@emotion/cache': 11.11.0
+ '@emotion/serialize': 1.1.2
+ '@emotion/use-insertion-effect-with-fallbacks': 1.0.1(react@18.2.0)
+ '@emotion/utils': 1.2.1
+ '@emotion/weak-memoize': 0.3.1
+ '@types/react': 18.2.6
+ hoist-non-react-statics: 3.3.2
+ react: 18.2.0
+ dev: false
+
+ /@emotion/serialize@1.1.2:
+ resolution: {integrity: sha512-zR6a/fkFP4EAcCMQtLOhIgpprZOwNmCldtpaISpvz348+DP4Mz8ZoKaGGCQpbzepNIUWbq4w6hNZkwDyKoS+HA==}
+ dependencies:
+ '@emotion/hash': 0.9.1
+ '@emotion/memoize': 0.8.1
+ '@emotion/unitless': 0.8.1
+ '@emotion/utils': 1.2.1
+ csstype: 3.1.2
+ dev: false
+
+ /@emotion/sheet@1.2.2:
+ resolution: {integrity: sha512-0QBtGvaqtWi+nx6doRwDdBIzhNdZrXUppvTM4dtZZWEGTXL/XE/yJxLMGlDT1Gt+UHH5IX1n+jkXyytE/av7OA==}
+ dev: false
+
+ /@emotion/styled@11.11.0(@emotion/react@11.11.0)(@types/react@18.2.6)(react@18.2.0):
+ resolution: {integrity: sha512-hM5Nnvu9P3midq5aaXj4I+lnSfNi7Pmd4EWk1fOZ3pxookaQTNew6bp4JaCBYM4HVFZF9g7UjJmsUmC2JlxOng==}
+ peerDependencies:
+ '@emotion/react': ^11.0.0-rc.0
+ '@types/react': '*'
+ react: '>=16.8.0'
+ peerDependenciesMeta:
+ '@types/react':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@emotion/babel-plugin': 11.11.0
+ '@emotion/is-prop-valid': 1.2.1
+ '@emotion/react': 11.11.0(@types/react@18.2.6)(react@18.2.0)
+ '@emotion/serialize': 1.1.2
+ '@emotion/use-insertion-effect-with-fallbacks': 1.0.1(react@18.2.0)
+ '@emotion/utils': 1.2.1
+ '@types/react': 18.2.6
+ react: 18.2.0
+ dev: false
+
+ /@emotion/unitless@0.8.1:
+ resolution: {integrity: sha512-KOEGMu6dmJZtpadb476IsZBclKvILjopjUii3V+7MnXIQCYh8W3NgNcgwo21n9LXZX6EDIKvqfjYxXebDwxKmQ==}
+ dev: false
+
+ /@emotion/use-insertion-effect-with-fallbacks@1.0.1(react@18.2.0):
+ resolution: {integrity: sha512-jT/qyKZ9rzLErtrjGgdkMBn2OP8wl0G3sQlBb3YPryvKHsjvINUhVaPFfP+fpBcOkmrVOVEEHQFJ7nbj2TH2gw==}
+ peerDependencies:
+ react: '>=16.8.0'
+ dependencies:
+ react: 18.2.0
+ dev: false
+
+ /@emotion/utils@1.2.1:
+ resolution: {integrity: sha512-Y2tGf3I+XVnajdItskUCn6LX+VUDmP6lTL4fcqsXAv43dnlbZiuW4MWQW38rW/BVWSE7Q/7+XQocmpnRYILUmg==}
+ dev: false
+
+ /@emotion/weak-memoize@0.3.1:
+ resolution: {integrity: sha512-EsBwpc7hBUJWAsNPBmJy4hxWx12v6bshQsldrVmjxJoc3isbxhOrF2IcCpaXxfvq03NwkI7sbsOLXbYuqF/8Ww==}
+ dev: false
+
+ /@esbuild/android-arm64@0.17.19:
+ resolution: {integrity: sha512-KBMWvEZooR7+kzY0BtbTQn0OAYY7CsiydT63pVEaPtVYF0hXbUaOyZog37DKxK7NF3XacBJOpYT4adIJh+avxA==}
+ engines: {node: '>=12'}
+ cpu: [arm64]
+ os: [android]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/android-arm@0.17.19:
+ resolution: {integrity: sha512-rIKddzqhmav7MSmoFCmDIb6e2W57geRsM94gV2l38fzhXMwq7hZoClug9USI2pFRGL06f4IOPHHpFNOkWieR8A==}
+ engines: {node: '>=12'}
+ cpu: [arm]
+ os: [android]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/android-x64@0.17.19:
+ resolution: {integrity: sha512-uUTTc4xGNDT7YSArp/zbtmbhO0uEEK9/ETW29Wk1thYUJBz3IVnvgEiEwEa9IeLyvnpKrWK64Utw2bgUmDveww==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [android]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/darwin-arm64@0.17.19:
+ resolution: {integrity: sha512-80wEoCfF/hFKM6WE1FyBHc9SfUblloAWx6FJkFWTWiCoht9Mc0ARGEM47e67W9rI09YoUxJL68WHfDRYEAvOhg==}
+ engines: {node: '>=12'}
+ cpu: [arm64]
+ os: [darwin]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/darwin-x64@0.17.19:
+ resolution: {integrity: sha512-IJM4JJsLhRYr9xdtLytPLSH9k/oxR3boaUIYiHkAawtwNOXKE8KoU8tMvryogdcT8AU+Bflmh81Xn6Q0vTZbQw==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [darwin]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/freebsd-arm64@0.17.19:
+ resolution: {integrity: sha512-pBwbc7DufluUeGdjSU5Si+P3SoMF5DQ/F/UmTSb8HXO80ZEAJmrykPyzo1IfNbAoaqw48YRpv8shwd1NoI0jcQ==}
+ engines: {node: '>=12'}
+ cpu: [arm64]
+ os: [freebsd]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/freebsd-x64@0.17.19:
+ resolution: {integrity: sha512-4lu+n8Wk0XlajEhbEffdy2xy53dpR06SlzvhGByyg36qJw6Kpfk7cp45DR/62aPH9mtJRmIyrXAS5UWBrJT6TQ==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [freebsd]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-arm64@0.17.19:
+ resolution: {integrity: sha512-ct1Tg3WGwd3P+oZYqic+YZF4snNl2bsnMKRkb3ozHmnM0dGWuxcPTTntAF6bOP0Sp4x0PjSF+4uHQ1xvxfRKqg==}
+ engines: {node: '>=12'}
+ cpu: [arm64]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-arm@0.17.19:
+ resolution: {integrity: sha512-cdmT3KxjlOQ/gZ2cjfrQOtmhG4HJs6hhvm3mWSRDPtZ/lP5oe8FWceS10JaSJC13GBd4eH/haHnqf7hhGNLerA==}
+ engines: {node: '>=12'}
+ cpu: [arm]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-ia32@0.17.19:
+ resolution: {integrity: sha512-w4IRhSy1VbsNxHRQpeGCHEmibqdTUx61Vc38APcsRbuVgK0OPEnQ0YD39Brymn96mOx48Y2laBQGqgZ0j9w6SQ==}
+ engines: {node: '>=12'}
+ cpu: [ia32]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-loong64@0.17.19:
+ resolution: {integrity: sha512-2iAngUbBPMq439a+z//gE+9WBldoMp1s5GWsUSgqHLzLJ9WoZLZhpwWuym0u0u/4XmZ3gpHmzV84PonE+9IIdQ==}
+ engines: {node: '>=12'}
+ cpu: [loong64]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-mips64el@0.17.19:
+ resolution: {integrity: sha512-LKJltc4LVdMKHsrFe4MGNPp0hqDFA1Wpt3jE1gEyM3nKUvOiO//9PheZZHfYRfYl6AwdTH4aTcXSqBerX0ml4A==}
+ engines: {node: '>=12'}
+ cpu: [mips64el]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-ppc64@0.17.19:
+ resolution: {integrity: sha512-/c/DGybs95WXNS8y3Ti/ytqETiW7EU44MEKuCAcpPto3YjQbyK3IQVKfF6nbghD7EcLUGl0NbiL5Rt5DMhn5tg==}
+ engines: {node: '>=12'}
+ cpu: [ppc64]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-riscv64@0.17.19:
+ resolution: {integrity: sha512-FC3nUAWhvFoutlhAkgHf8f5HwFWUL6bYdvLc/TTuxKlvLi3+pPzdZiFKSWz/PF30TB1K19SuCxDTI5KcqASJqA==}
+ engines: {node: '>=12'}
+ cpu: [riscv64]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-s390x@0.17.19:
+ resolution: {integrity: sha512-IbFsFbxMWLuKEbH+7sTkKzL6NJmG2vRyy6K7JJo55w+8xDk7RElYn6xvXtDW8HCfoKBFK69f3pgBJSUSQPr+4Q==}
+ engines: {node: '>=12'}
+ cpu: [s390x]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/linux-x64@0.17.19:
+ resolution: {integrity: sha512-68ngA9lg2H6zkZcyp22tsVt38mlhWde8l3eJLWkyLrp4HwMUr3c1s/M2t7+kHIhvMjglIBrFpncX1SzMckomGw==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [linux]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/netbsd-x64@0.17.19:
+ resolution: {integrity: sha512-CwFq42rXCR8TYIjIfpXCbRX0rp1jo6cPIUPSaWwzbVI4aOfX96OXY8M6KNmtPcg7QjYeDmN+DD0Wp3LaBOLf4Q==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [netbsd]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/openbsd-x64@0.17.19:
+ resolution: {integrity: sha512-cnq5brJYrSZ2CF6c35eCmviIN3k3RczmHz8eYaVlNasVqsNY+JKohZU5MKmaOI+KkllCdzOKKdPs762VCPC20g==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [openbsd]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/sunos-x64@0.17.19:
+ resolution: {integrity: sha512-vCRT7yP3zX+bKWFeP/zdS6SqdWB8OIpaRq/mbXQxTGHnIxspRtigpkUcDMlSCOejlHowLqII7K2JKevwyRP2rg==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [sunos]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/win32-arm64@0.17.19:
+ resolution: {integrity: sha512-yYx+8jwowUstVdorcMdNlzklLYhPxjniHWFKgRqH7IFlUEa0Umu3KuYplf1HUZZ422e3NU9F4LGb+4O0Kdcaag==}
+ engines: {node: '>=12'}
+ cpu: [arm64]
+ os: [win32]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/win32-ia32@0.17.19:
+ resolution: {integrity: sha512-eggDKanJszUtCdlVs0RB+h35wNlb5v4TWEkq4vZcmVt5u/HiDZrTXe2bWFQUez3RgNHwx/x4sk5++4NSSicKkw==}
+ engines: {node: '>=12'}
+ cpu: [ia32]
+ os: [win32]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@esbuild/win32-x64@0.17.19:
+ resolution: {integrity: sha512-lAhycmKnVOuRYNtRtatQR1LPQf2oYCkRGkSFnseDAKPl8lu5SOsK/e1sXe5a0Pc5kHIHe6P2I/ilntNv2xf3cA==}
+ engines: {node: '>=12'}
+ cpu: [x64]
+ os: [win32]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /@eslint-community/eslint-utils@4.4.0(eslint@8.41.0):
+ resolution: {integrity: sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==}
+ engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
+ peerDependencies:
+ eslint: ^6.0.0 || ^7.0.0 || >=8.0.0
+ dependencies:
+ eslint: 8.41.0
+ eslint-visitor-keys: 3.4.1
+ dev: true
+
+ /@eslint-community/regexpp@4.5.1:
+ resolution: {integrity: sha512-Z5ba73P98O1KUYCCJTUeVpja9RcGoMdncZ6T49FCUl2lN38JtCJ+3WgIDBv0AuY4WChU5PmtJmOCTlN6FZTFKQ==}
+ engines: {node: ^12.0.0 || ^14.0.0 || >=16.0.0}
+ dev: true
+
+ /@eslint/eslintrc@2.0.3:
+ resolution: {integrity: sha512-+5gy6OQfk+xx3q0d6jGZZC3f3KzAkXc/IanVxd1is/VIIziRqqt3ongQz0FiTUXqTk0c7aDB3OaFuKnuSoJicQ==}
+ engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
+ dependencies:
+ ajv: 6.12.6
+ debug: 4.3.4
+ espree: 9.5.2
+ globals: 13.20.0
+ ignore: 5.2.4
+ import-fresh: 3.3.0
+ js-yaml: 4.1.0
+ minimatch: 3.1.2
+ strip-json-comments: 3.1.1
+ transitivePeerDependencies:
+ - supports-color
+ dev: true
+
+ /@eslint/js@8.41.0:
+ resolution: {integrity: sha512-LxcyMGxwmTh2lY9FwHPGWOHmYFCZvbrFCBZL4FzSSsxsRPuhrYUg/49/0KDfW8tnIEaEHtfmn6+NPN+1DqaNmA==}
+ engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
+ dev: true
+
+ /@humanwhocodes/config-array@0.11.8:
+ resolution: {integrity: sha512-UybHIJzJnR5Qc/MsD9Kr+RpO2h+/P1GhOwdiLPXK5TWk5sgTdu88bTD9UP+CKbPPh5Rni1u0GjAdYQLemG8g+g==}
+ engines: {node: '>=10.10.0'}
+ dependencies:
+ '@humanwhocodes/object-schema': 1.2.1
+ debug: 4.3.4
+ minimatch: 3.1.2
+ transitivePeerDependencies:
+ - supports-color
+ dev: true
+
+ /@humanwhocodes/module-importer@1.0.1:
+ resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==}
+ engines: {node: '>=12.22'}
+ dev: true
+
+ /@humanwhocodes/object-schema@1.2.1:
+ resolution: {integrity: sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==}
+ dev: true
+
+ /@jest/expect-utils@29.5.0:
+ resolution: {integrity: sha512-fmKzsidoXQT2KwnrwE0SQq3uj8Z763vzR8LnLBwC2qYWEFpjX8daRsk6rHUM1QvNlEW/UJXNXm59ztmJJWs2Mg==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ jest-get-type: 29.4.3
+ dev: true
+
+ /@jest/schemas@29.4.3:
+ resolution: {integrity: sha512-VLYKXQmtmuEz6IxJsrZwzG9NvtkQsWNnWMsKxqWNu3+CnfzJQhp0WDDKWLVV9hLKr0l3SLLFRqcYHjhtyuDVxg==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ '@sinclair/typebox': 0.25.24
+ dev: true
+
+ /@jest/types@29.5.0:
+ resolution: {integrity: sha512-qbu7kN6czmVRc3xWFQcAN03RAUamgppVUdXrvl1Wr3jlNF93o9mJbGcDWrwGB6ht44u7efB1qCFgVQmca24Uog==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ '@jest/schemas': 29.4.3
+ '@types/istanbul-lib-coverage': 2.0.4
+ '@types/istanbul-reports': 3.0.1
+ '@types/node': 20.2.3
+ '@types/yargs': 17.0.24
+ chalk: 4.1.2
+ dev: true
+
+ /@jridgewell/gen-mapping@0.3.3:
+ resolution: {integrity: sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ==}
+ engines: {node: '>=6.0.0'}
+ dependencies:
+ '@jridgewell/set-array': 1.1.2
+ '@jridgewell/sourcemap-codec': 1.4.15
+ '@jridgewell/trace-mapping': 0.3.18
+
+ /@jridgewell/resolve-uri@3.1.0:
+ resolution: {integrity: sha512-F2msla3tad+Mfht5cJq7LSXcdudKTWCVYUgw6pLFOOHSTtZlj6SWNYAp+AhuqLmWdBO2X5hPrLcu8cVP8fy28w==}
+ engines: {node: '>=6.0.0'}
+
+ /@jridgewell/set-array@1.1.2:
+ resolution: {integrity: sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw==}
+ engines: {node: '>=6.0.0'}
+
+ /@jridgewell/sourcemap-codec@1.4.14:
+ resolution: {integrity: sha512-XPSJHWmi394fuUuzDnGz1wiKqWfo1yXecHQMRf2l6hztTO+nPru658AyDngaBe7isIxEkRsPR3FZh+s7iVa4Uw==}
+
+ /@jridgewell/sourcemap-codec@1.4.15:
+ resolution: {integrity: sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==}
+
+ /@jridgewell/trace-mapping@0.3.18:
+ resolution: {integrity: sha512-w+niJYzMHdd7USdiH2U6869nqhD2nbfZXND5Yp93qIbEmnDNk7PD48o+YchRVpzMU7M6jVCbenTR7PA1FLQ9pA==}
+ dependencies:
+ '@jridgewell/resolve-uri': 3.1.0
+ '@jridgewell/sourcemap-codec': 1.4.14
+
+ /@linaria/core@4.2.9:
+ resolution: {integrity: sha512-ELcu37VNVOT/PU0L6WDIN+aLzNFyJrqoBYT0CucGOCAmODbojUMCv8oJYRbWzA3N34w1t199dN4UFdfRWFG2rg==}
+ engines: {node: ^12.16.0 || >=13.7.0}
+ dependencies:
+ '@linaria/logger': 4.0.0
+ '@linaria/tags': 4.3.5
+ '@linaria/utils': 4.3.4
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /@linaria/logger@4.0.0:
+ resolution: {integrity: sha512-YnBq0JlDWMEkTOK+tMo5yEVR0f5V//6qMLToGcLhTyM9g9i+IDFn51Z+5q2hLk7RdG4NBPgbcCXYi2w4RKsPeg==}
+ engines: {node: ^12.16.0 || >=13.7.0}
+ dependencies:
+ debug: 4.3.4
+ picocolors: 1.0.0
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /@linaria/tags@4.3.5:
+ resolution: {integrity: sha512-PgaIi8Vv89YOjc6rpKL/uPg2w4k0rAwAYxcqeXqzKqsEAste5rgB8xp1/KUOG0oAOkPd3MRL6Duj+m0ZwJ3g+g==}
+ engines: {node: ^12.16.0 || >=13.7.0}
+ dependencies:
+ '@babel/generator': 7.21.5
+ '@linaria/logger': 4.0.0
+ '@linaria/utils': 4.3.4
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /@linaria/utils@4.3.4:
+ resolution: {integrity: sha512-vt6WJG54n+KANaqxOfzIIU7aSfFHEWFbnGLsgxL7nASHqO0zezrNA2y2Rrp80zSeTW+wSpbmDM4uJyC9UW1qoA==}
+ engines: {node: ^12.16.0 || >=13.7.0}
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/plugin-proposal-export-namespace-from': 7.18.9(@babel/core@7.21.8)
+ '@babel/plugin-syntax-dynamic-import': 7.8.3(@babel/core@7.21.8)
+ '@babel/plugin-transform-modules-commonjs': 7.21.5(@babel/core@7.21.8)
+ '@babel/traverse': 7.21.5
+ '@babel/types': 7.21.5
+ '@linaria/logger': 4.0.0
+ babel-merge: 3.0.0(@babel/core@7.21.8)
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /@mui/base@5.0.0-beta.1(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-xrkDCeu3JQE+JjJUnJnOrdQJMXwKhbV4AW+FRjMIj5i9cHK3BAuatG/iqbf1M+jklVWLk0KdbgioKwK+03aYbA==}
+ engines: {node: '>=12.0.0'}
+ peerDependencies:
+ '@types/react': ^17.0.0 || ^18.0.0
+ react: ^17.0.0 || ^18.0.0
+ react-dom: ^17.0.0 || ^18.0.0
+ peerDependenciesMeta:
+ '@types/react':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@emotion/is-prop-valid': 1.2.1
+ '@mui/types': 7.2.4(@types/react@18.2.6)
+ '@mui/utils': 5.13.1(react@18.2.0)
+ '@popperjs/core': 2.11.7
+ '@types/react': 18.2.6
+ clsx: 1.2.1
+ prop-types: 15.8.1
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ react-is: 18.2.0
+ dev: false
+
+ /@mui/core-downloads-tracker@5.13.1:
+ resolution: {integrity: sha512-qDHtNDO72NcBQMhaWBt9EZMvNiO+OXjPg5Sdk/6LgRDw6Zr3HdEZ5n2FJ/qtYsaT/okGyCuQavQkcZCOCEVf/g==}
+ dev: false
+
+ /@mui/icons-material@5.11.16(@mui/material@5.13.1)(@types/react@18.2.6)(react@18.2.0):
+ resolution: {integrity: sha512-oKkx9z9Kwg40NtcIajF9uOXhxiyTZrrm9nmIJ4UjkU2IdHpd4QVLbCc/5hZN/y0C6qzi2Zlxyr9TGddQx2vx2A==}
+ engines: {node: '>=12.0.0'}
+ peerDependencies:
+ '@mui/material': ^5.0.0
+ '@types/react': ^17.0.0 || ^18.0.0
+ react: ^17.0.0 || ^18.0.0
+ peerDependenciesMeta:
+ '@types/react':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@mui/material': 5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0)
+ '@types/react': 18.2.6
+ react: 18.2.0
+ dev: false
+
+ /@mui/material@5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-qSnbJZer8lIuDYFDv19/t3s0AXYY9SxcOdhCnGvetRSfOG4gy3TkiFXNCdW5OLNveTieiMpOuv46eXUmE3ZA6A==}
+ engines: {node: '>=12.0.0'}
+ peerDependencies:
+ '@emotion/react': ^11.5.0
+ '@emotion/styled': ^11.3.0
+ '@types/react': ^17.0.0 || ^18.0.0
+ react: ^17.0.0 || ^18.0.0
+ react-dom: ^17.0.0 || ^18.0.0
+ peerDependenciesMeta:
+ '@emotion/react':
+ optional: true
+ '@emotion/styled':
+ optional: true
+ '@types/react':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@emotion/react': 11.11.0(@types/react@18.2.6)(react@18.2.0)
+ '@emotion/styled': 11.11.0(@emotion/react@11.11.0)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/base': 5.0.0-beta.1(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0)
+ '@mui/core-downloads-tracker': 5.13.1
+ '@mui/system': 5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/types': 7.2.4(@types/react@18.2.6)
+ '@mui/utils': 5.13.1(react@18.2.0)
+ '@types/react': 18.2.6
+ '@types/react-transition-group': 4.4.6
+ clsx: 1.2.1
+ csstype: 3.1.2
+ prop-types: 15.8.1
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ react-is: 18.2.0
+ react-transition-group: 4.4.5(react-dom@18.2.0)(react@18.2.0)
+ dev: false
+
+ /@mui/private-theming@5.13.1(@types/react@18.2.6)(react@18.2.0):
+ resolution: {integrity: sha512-HW4npLUD9BAkVppOUZHeO1FOKUJWAwbpy0VQoGe3McUYTlck1HezGHQCfBQ5S/Nszi7EViqiimECVl9xi+/WjQ==}
+ engines: {node: '>=12.0.0'}
+ peerDependencies:
+ '@types/react': ^17.0.0 || ^18.0.0
+ react: ^17.0.0 || ^18.0.0
+ peerDependenciesMeta:
+ '@types/react':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@mui/utils': 5.13.1(react@18.2.0)
+ '@types/react': 18.2.6
+ prop-types: 15.8.1
+ react: 18.2.0
+ dev: false
+
+ /@mui/styled-engine@5.12.3(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(react@18.2.0):
+ resolution: {integrity: sha512-AhZtiRyT8Bjr7fufxE/mLS+QJ3LxwX1kghIcM2B2dvJzSSg9rnIuXDXM959QfUVIM3C8U4x3mgVoPFMQJvc4/g==}
+ engines: {node: '>=12.0.0'}
+ peerDependencies:
+ '@emotion/react': ^11.4.1
+ '@emotion/styled': ^11.3.0
+ react: ^17.0.0 || ^18.0.0
+ peerDependenciesMeta:
+ '@emotion/react':
+ optional: true
+ '@emotion/styled':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@emotion/cache': 11.11.0
+ '@emotion/react': 11.11.0(@types/react@18.2.6)(react@18.2.0)
+ '@emotion/styled': 11.11.0(@emotion/react@11.11.0)(@types/react@18.2.6)(react@18.2.0)
+ csstype: 3.1.2
+ prop-types: 15.8.1
+ react: 18.2.0
+ dev: false
+
+ /@mui/system@5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react@18.2.0):
+ resolution: {integrity: sha512-BsDUjhiO6ZVAvzKhnWBHLZ5AtPJcdT+62VjnRLyA4isboqDKLg4fmYIZXq51yndg/soDK9RkY5lYZwEDku13Ow==}
+ engines: {node: '>=12.0.0'}
+ peerDependencies:
+ '@emotion/react': ^11.5.0
+ '@emotion/styled': ^11.3.0
+ '@types/react': ^17.0.0 || ^18.0.0
+ react: ^17.0.0 || ^18.0.0
+ peerDependenciesMeta:
+ '@emotion/react':
+ optional: true
+ '@emotion/styled':
+ optional: true
+ '@types/react':
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@emotion/react': 11.11.0(@types/react@18.2.6)(react@18.2.0)
+ '@emotion/styled': 11.11.0(@emotion/react@11.11.0)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/private-theming': 5.13.1(@types/react@18.2.6)(react@18.2.0)
+ '@mui/styled-engine': 5.12.3(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(react@18.2.0)
+ '@mui/types': 7.2.4(@types/react@18.2.6)
+ '@mui/utils': 5.13.1(react@18.2.0)
+ '@types/react': 18.2.6
+ clsx: 1.2.1
+ csstype: 3.1.2
+ prop-types: 15.8.1
+ react: 18.2.0
+ dev: false
+
+ /@mui/types@7.2.4(@types/react@18.2.6):
+ resolution: {integrity: sha512-LBcwa8rN84bKF+f5sDyku42w1NTxaPgPyYKODsh01U1fVstTClbUoSA96oyRBnSNyEiAVjKm6Gwx9vjR+xyqHA==}
+ peerDependencies:
+ '@types/react': '*'
+ peerDependenciesMeta:
+ '@types/react':
+ optional: true
+ dependencies:
+ '@types/react': 18.2.6
+ dev: false
+
+ /@mui/utils@5.13.1(react@18.2.0):
+ resolution: {integrity: sha512-6lXdWwmlUbEU2jUI8blw38Kt+3ly7xkmV9ljzY4Q20WhsJMWiNry9CX8M+TaP/HbtuyR8XKsdMgQW7h7MM3n3A==}
+ engines: {node: '>=12.0.0'}
+ peerDependencies:
+ react: ^17.0.0 || ^18.0.0
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@types/prop-types': 15.7.5
+ '@types/react-is': 18.2.0
+ prop-types: 15.8.1
+ react: 18.2.0
+ react-is: 18.2.0
+ dev: false
+
+ /@nodelib/fs.scandir@2.1.5:
+ resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==}
+ engines: {node: '>= 8'}
+ dependencies:
+ '@nodelib/fs.stat': 2.0.5
+ run-parallel: 1.2.0
+ dev: true
+
+ /@nodelib/fs.stat@2.0.5:
+ resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==}
+ engines: {node: '>= 8'}
+ dev: true
+
+ /@nodelib/fs.walk@1.2.8:
+ resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==}
+ engines: {node: '>= 8'}
+ dependencies:
+ '@nodelib/fs.scandir': 2.1.5
+ fastq: 1.15.0
+ dev: true
+
+ /@popperjs/core@2.11.7:
+ resolution: {integrity: sha512-Cr4OjIkipTtcXKjAsm8agyleBuDHvxzeBoa1v543lbv1YaIwQjESsVcmjiWiPEbC1FIeHOG/Op9kdCmAmiS3Kw==}
+ dev: false
+
+ /@remirror/core-constants@2.0.1:
+ resolution: {integrity: sha512-ZR4aihtnnT9lMbhh5DEbsriJRlukRXmLZe7HmM+6ufJNNUDoazc75UX26xbgQlNUqgAqMcUdGFAnPc1JwgAdLQ==}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ dev: false
+
+ /@remirror/core-helpers@2.0.3:
+ resolution: {integrity: sha512-LqIPF4stGG69l9qu/FFicv9d9B+YaItzgDMC5A0CEvDQfKkGD3BfabLmfpnuWbsc06oKGdTduilgWcALLZoYLg==}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@linaria/core': 4.2.9
+ '@remirror/core-constants': 2.0.1
+ '@remirror/types': 1.0.1
+ '@types/object.omit': 3.0.0
+ '@types/object.pick': 1.3.2
+ '@types/throttle-debounce': 2.1.0
+ case-anything: 2.1.11
+ dash-get: 1.0.2
+ deepmerge: 4.3.1
+ fast-deep-equal: 3.1.3
+ make-error: 1.3.6
+ object.omit: 3.0.0
+ object.pick: 1.3.0
+ throttle-debounce: 3.0.1
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /@remirror/types@1.0.1:
+ resolution: {integrity: sha512-VlZQxwGnt1jtQ18D6JqdIF+uFZo525WEqrfp9BOc3COPpK4+AWCgdnAWL+ho6imWcoINlGjR/+3b6y5C1vBVEA==}
+ dependencies:
+ type-fest: 2.19.0
+ dev: false
+
+ /@remix-run/router@1.6.2:
+ resolution: {integrity: sha512-LzqpSrMK/3JBAVBI9u3NWtOhWNw5AMQfrUFYB0+bDHTSw17z++WJLsPsxAuK+oSddsxk4d7F/JcdDPM1M5YAhA==}
+ engines: {node: '>=14'}
+ dev: false
+
+ /@sinclair/typebox@0.25.24:
+ resolution: {integrity: sha512-XJfwUVUKDHF5ugKwIcxEgc9k8b7HbznCp6eUfWgu710hMPNIO4aw4/zB5RogDQz8nd6gyCDpU9O/m6qYEWY6yQ==}
+ dev: true
+
+ /@testing-library/dom@9.3.0:
+ resolution: {integrity: sha512-Dffe68pGwI6WlLRYR2I0piIkyole9cSBH5jGQKCGMRpHW5RHCqAUaqc2Kv0tUyd4dU4DLPKhJIjyKOnjv4tuUw==}
+ engines: {node: '>=14'}
+ dependencies:
+ '@babel/code-frame': 7.21.4
+ '@babel/runtime': 7.21.5
+ '@types/aria-query': 5.0.1
+ aria-query: 5.1.3
+ chalk: 4.1.2
+ dom-accessibility-api: 0.5.16
+ lz-string: 1.5.0
+ pretty-format: 27.5.1
+ dev: true
+
+ /@testing-library/jest-dom@5.16.5:
+ resolution: {integrity: sha512-N5ixQ2qKpi5OLYfwQmUb/5mSV9LneAcaUfp32pn4yCnpb8r/Yz0pXFPck21dIicKmi+ta5WRAknkZCfA8refMA==}
+ engines: {node: '>=8', npm: '>=6', yarn: '>=1'}
+ dependencies:
+ '@adobe/css-tools': 4.2.0
+ '@babel/runtime': 7.21.5
+ '@types/testing-library__jest-dom': 5.14.5
+ aria-query: 5.1.3
+ chalk: 3.0.0
+ css.escape: 1.5.1
+ dom-accessibility-api: 0.5.16
+ lodash: 4.17.21
+ redent: 3.0.0
+ dev: true
+
+ /@testing-library/react@14.0.0(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-S04gSNJbYE30TlIMLTzv6QCTzt9AqIF5y6s6SzVFILNcNvbV/jU96GeiTPillGQo+Ny64M/5PV7klNYYgv5Dfg==}
+ engines: {node: '>=14'}
+ peerDependencies:
+ react: ^18.0.0
+ react-dom: ^18.0.0
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@testing-library/dom': 9.3.0
+ '@types/react-dom': 18.2.4
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ dev: true
+
+ /@testing-library/user-event@14.4.3(@testing-library/dom@9.3.0):
+ resolution: {integrity: sha512-kCUc5MEwaEMakkO5x7aoD+DLi02ehmEM2QCGWvNqAS1dV/fAvORWEjnjsEIvml59M7Y5kCkWN6fCCyPOe8OL6Q==}
+ engines: {node: '>=12', npm: '>=6'}
+ peerDependencies:
+ '@testing-library/dom': '>=7.21.4'
+ dependencies:
+ '@testing-library/dom': 9.3.0
+ dev: true
+
+ /@tiptap/core@2.0.3(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-jLyVIWAdjjlNzrsRhSE2lVL/7N8228/1R1QtaVU85UlMIwHFAcdzhD8FeiKkqxpTnGpaDVaTy7VNEtEgaYdCyA==}
+ peerDependencies:
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-blockquote@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-rkUcFv2iL6f86DBBHoa4XdKNG2StvkJ7tfY9GoMpT46k3nxOaMTqak9/qZOo79TWxMLYtXzoxtKIkmWsbbcj4A==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-bold@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-OGT62fMRovSSayjehumygFWTg2Qn0IDbqyMpigg/RUAsnoOI2yBZFVrdM2gk1StyoSay7gTn2MLw97IUfr7FXg==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-bubble-menu@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-lPt1ELrYCuoQrQEUukqjp9xt38EwgPUwaKHI3wwt2Rbv+C6q1gmRsK1yeO/KqCNmFxNqF2p9ZF9srOnug/RZDQ==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ tippy.js: 6.3.7
+ dev: false
+
+ /@tiptap/extension-bullet-list@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-RtaLiRvZbMTOje+FW5bn+mYogiIgNxOm065wmyLPypnTbLSeHeYkoqVSqzZeqUn+7GLnwgn1shirUe6csVE/BA==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-code-block@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-F4xMy18EwgpyY9f5Te7UuF7UwxRLptOtCq1p2c2DfxBvHDWhAjQqVqcW/sq/I/WuED7FwCnPLyyAasPiVPkLPw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-code@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-LsVCKVxgBtkstAr1FjxN8T3OjlC76a2X8ouoZpELMp+aXbjqyanCKzt+sjjUhE4H0yLFd4v+5v6UFoCv4EILiw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-color@2.0.3(@tiptap/core@2.0.3)(@tiptap/extension-text-style@2.0.3):
+ resolution: {integrity: sha512-LYj3CWahhuJOy4/bwOur+cob8eky7xx7wyyBFIYELuzLcZt9hBmZwXxinQzD7BaQv4YdT+3oqr8BhChuPNj52w==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/extension-text-style': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/extension-text-style': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-document@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-PsYeNQQBYIU9ayz1R11Kv/kKNPFNIV8tApJ9pxelXjzcAhkjncNUazPN/dyho60mzo+WpsmS3ceTj/gK3bCtWA==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-dropcursor@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-McthMrfusn6PjcaynJLheZJcXto8TaIW5iVitYh8qQrDXr31MALC/5GvWuiswmQ8bAXiWPwlLDYE/OJfwtggaw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-floating-menu@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-zN1vRGRvyK3pO2aHRmQSOTpl4UJraXYwKYM009n6WviYKUNm0LPGo+VD4OAtdzUhPXyccnlsTv2p6LIqFty6Bg==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ tippy.js: 6.3.7
+ dev: false
+
+ /@tiptap/extension-gapcursor@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-6I9EzzsYOyyqDvDvxIK6Rv3EXB+fHKFj8ntHO8IXmeNJ6pkhOinuXVsW6Yo7TcDYoTj4D5I2MNFAW2rIkgassw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-hard-break@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-RCln6ARn16jvKTjhkcAD5KzYXYS0xRMc0/LrHeV8TKdCd4Yd0YYHe0PU4F9gAgAfPQn7Dgt4uTVJLN11ICl8sQ==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-heading@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-f0IEv5ms6aCzL80WeZ1qLCXTkRVwbpRr1qAETjg3gG4eoJN18+lZNOJYpyZy3P92C5KwF2T3Av00eFyVLIbb8Q==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-highlight@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-NrtibY8cZkIjZMQuHRrKd4php+plOvAoSo8g3uVFu275I/Ixt5HqJ53R4voCXs8W8BOBRs2HS2QX8Cjh79XhtA==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-history@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-00KHIcJ8kivn2ARI6NQYphv2LfllVCXViHGm0EhzDW6NQxCrriJKE3tKDcTFCu7LlC5doMpq9Z6KXdljc4oVeQ==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-horizontal-rule@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-SZRUSh07b/M0kJHNKnfBwBMWrZBEm/E2LrK1NbluwT3DBhE+gvwiEdBxgB32zKHNxaDEXUJwUIPNC3JSbKvPUA==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-image@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-hS9ZJwz0md07EHsC+o4NuuJkhCZsZn7TuRz/2CvRSj2fWFIz+40CyNAHf/2J0qNugG9ommXaemetsADeEZP9ag==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-italic@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-cfS5sW0gu7qf4ihwnLtW/QMTBrBEXaT0sJl3RwkhjIBg/65ywJKE5Nz9ewnQHmDeT18hvMJJ1VIb4j4ze9jj9A==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-link@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-H72tXQ5rkVCkAhFaf08fbEU7EBUCK0uocsqOF+4th9sOlrhfgyJtc8Jv5EXPDpxNgG5jixSqWBo0zKXQm9s9eg==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ linkifyjs: 4.1.1
+ dev: false
+
+ /@tiptap/extension-list-item@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-p7cUsk0LpM1PfdAuFE8wYBNJ3gvA0UhNGR08Lo++rt9UaCeFLSN1SXRxg97c0oa5+Ski7SrCjIJ5Ynhz0viTjQ==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-ordered-list@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-ZB3MpZh/GEy1zKgw7XDQF4FIwycZWNof1k9WbDZOI063Ch4qHZowhVttH2mTCELuyvTMM/o9a8CS7qMqQB48bw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-paragraph@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-a+tKtmj4bU3GVCH1NE8VHWnhVexxX5boTVxsHIr4yGG3UoKo1c5AO7YMaeX2W5xB5iIA+BQqOPCDPEAx34dd2A==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-placeholder@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-Z42jo0termRAf0S0L8oxrts94IWX5waU4isS2CUw8xCUigYyCFslkhQXkWATO1qRbjNFLKN2C9qvCgGf4UeBrw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ dev: false
+
+ /@tiptap/extension-strike@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-RO4/EYe2iPD6ifDHORT8fF6O9tfdtnzxLGwZIKZXnEgtweH+MgoqevEzXYdS+54Wraq4TUQGNcsYhe49pv7Rlw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-text-align@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-VlLgqncKdjMjVjbU60/ALYhFs0wUdjAyvjDXnH1OoM/HuzbILvufPMYz4DUieJIWVJOYUKHQgg4XwBWceAM2Tw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-text-style@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-yHIYtZVewSwfBfI6TffnsDRiOuXzytppcCsaDlsZFm8OtLG8v9ioH0ItMoOstmZZBiWJOm8iOy2yWSc4rNQEJw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-text@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-LvzChcTCcPSMNLUjZe/A9SHXWGDHtvk73fR7CBqAeNU0MxhBPEBI03GFQ6RzW3xX0CmDmjpZoDxFMB+hDEtW1A==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/extension-underline@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-oMYa7qib/5wJjpUp79GZEe+E/iyf1oZBsgiG26IspEtVTHZmpn3+Ktud7l43y/hpTeEzFTKOF1/uVbayHtSERg==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ dev: false
+
+ /@tiptap/pm@2.0.3(@tiptap/core@2.0.3):
+ resolution: {integrity: sha512-I9dsInD89Agdm1QjFRO9dmJtU1ldVSILNPW0pEhv9wYqYVvl4HUj/JMtYNqu2jWrCHNXQcaX/WkdSdvGJtmg5g==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ prosemirror-changeset: 2.2.1
+ prosemirror-collab: 1.3.1
+ prosemirror-commands: 1.5.2
+ prosemirror-dropcursor: 1.8.1
+ prosemirror-gapcursor: 1.3.2
+ prosemirror-history: 1.3.2
+ prosemirror-inputrules: 1.2.1
+ prosemirror-keymap: 1.2.2
+ prosemirror-markdown: 1.11.0
+ prosemirror-menu: 1.2.2
+ prosemirror-model: 1.19.1
+ prosemirror-schema-basic: 1.2.2
+ prosemirror-schema-list: 1.2.3
+ prosemirror-state: 1.4.3
+ prosemirror-tables: 1.3.2
+ prosemirror-trailing-node: 2.0.4(prosemirror-model@1.19.1)(prosemirror-state@1.4.3)(prosemirror-view@1.31.3)
+ prosemirror-transform: 1.7.2
+ prosemirror-view: 1.31.3
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /@tiptap/react@2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-fiAh8Lk+/NBPAR/PE4Kc/aLiBUbUYI/CpAopz8DI9eInNyV8h8LAGa9uFILJQF/TNu0tclJ4rV0sWc7Se0FZMw==}
+ peerDependencies:
+ '@tiptap/core': ^2.0.0
+ '@tiptap/pm': ^2.0.0
+ react: ^17.0.0 || ^18.0.0
+ react-dom: ^17.0.0 || ^18.0.0
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/extension-bubble-menu': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-floating-menu': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ dev: false
+
+ /@tiptap/starter-kit@2.0.3(@tiptap/pm@2.0.3):
+ resolution: {integrity: sha512-t4WG4w93zTpL2VxhVyJJvl3kdLF001ZrhpOuEiZqEMBMUMbM56Uiigv1CnUQpTFrjDAh3IM8hkqzAh20TYw2iQ==}
+ dependencies:
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/extension-blockquote': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-bold': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-bullet-list': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-code': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-code-block': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-document': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-dropcursor': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-gapcursor': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-hard-break': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-heading': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-history': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-horizontal-rule': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-italic': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-list-item': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-ordered-list': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-paragraph': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-strike': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-text': 2.0.3(@tiptap/core@2.0.3)
+ transitivePeerDependencies:
+ - '@tiptap/pm'
+ dev: false
+
+ /@types/aria-query@5.0.1:
+ resolution: {integrity: sha512-XTIieEY+gvJ39ChLcB4If5zHtPxt3Syj5rgZR+e1ctpmK8NjPf0zFqsz4JpLJT0xla9GFDKjy8Cpu331nrmE1Q==}
+ dev: true
+
+ /@types/history@4.7.11:
+ resolution: {integrity: sha512-qjDJRrmvBMiTx+jyLxvLfJU7UznFuokDv4f3WRuriHKERccVpFU+8XMQUAbDzoiJCsmexxRExQeMwwCdamSKDA==}
+ dev: true
+
+ /@types/istanbul-lib-coverage@2.0.4:
+ resolution: {integrity: sha512-z/QT1XN4K4KYuslS23k62yDIDLwLFkzxOuMplDtObz0+y7VqJCaO2o+SPwHCvLFZh7xazvvoor2tA/hPz9ee7g==}
+ dev: true
+
+ /@types/istanbul-lib-report@3.0.0:
+ resolution: {integrity: sha512-plGgXAPfVKFoYfa9NpYDAkseG+g6Jr294RqeqcqDixSbU34MZVJRi/P+7Y8GDpzkEwLaGZZOpKIEmeVZNtKsrg==}
+ dependencies:
+ '@types/istanbul-lib-coverage': 2.0.4
+ dev: true
+
+ /@types/istanbul-reports@3.0.1:
+ resolution: {integrity: sha512-c3mAZEuK0lvBp8tmuL74XRKn1+y2dcwOUpH7x4WrF6gk1GIgiluDRgMYQtw2OFcBvAJWlt6ASU3tSqxp0Uu0Aw==}
+ dependencies:
+ '@types/istanbul-lib-report': 3.0.0
+ dev: true
+
+ /@types/jest@29.5.1:
+ resolution: {integrity: sha512-tEuVcHrpaixS36w7hpsfLBLpjtMRJUE09/MHXn923LOVojDwyC14cWcfc0rDs0VEfUyYmt/+iX1kxxp+gZMcaQ==}
+ dependencies:
+ expect: 29.5.0
+ pretty-format: 29.5.0
+ dev: true
+
+ /@types/json5@0.0.29:
+ resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==}
+ dev: true
+
+ /@types/node@20.2.3:
+ resolution: {integrity: sha512-pg9d0yC4rVNWQzX8U7xb4olIOFuuVL9za3bzMT2pu2SU0SNEi66i2qrvhE2qt0HvkhuCaWJu7pLNOt/Pj8BIrw==}
+ dev: true
+
+ /@types/object.omit@3.0.0:
+ resolution: {integrity: sha512-I27IoPpH250TUzc9FzXd0P1BV/BMJuzqD3jOz98ehf9dQqGkxlq+hO1bIqZGWqCg5bVOy0g4AUVJtnxe0klDmw==}
+ dev: false
+
+ /@types/object.pick@1.3.2:
+ resolution: {integrity: sha512-sn7L+qQ6RLPdXRoiaE7bZ/Ek+o4uICma/lBFPyJEKDTPTBP1W8u0c4baj3EiS4DiqLs+Hk+KUGvMVJtAw3ePJg==}
+ dev: false
+
+ /@types/parse-json@4.0.0:
+ resolution: {integrity: sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==}
+ dev: false
+
+ /@types/prop-types@15.7.5:
+ resolution: {integrity: sha512-JCB8C6SnDoQf0cNycqd/35A7MjcnK+ZTqE7judS6o7utxUCg6imJg3QK2qzHKszlTjcj2cn+NwMB2i96ubpj7w==}
+
+ /@types/react-dom@18.2.4:
+ resolution: {integrity: sha512-G2mHoTMTL4yoydITgOGwWdWMVd8sNgyEP85xVmMKAPUBwQWm9wBPQUmvbeF4V3WBY1P7mmL4BkjQ0SqUpf1snw==}
+ dependencies:
+ '@types/react': 18.2.6
+ dev: true
+
+ /@types/react-is@18.2.0:
+ resolution: {integrity: sha512-1vz2yObaQkLL7YFe/pme2cpvDsCwI1WXIfL+5eLz0MI9gFG24Re16RzUsI8t9XZn9ZWvgLNDrJBmrqXJO7GNQQ==}
+ dependencies:
+ '@types/react': 18.2.6
+ dev: false
+
+ /@types/react-router-dom@5.3.3:
+ resolution: {integrity: sha512-kpqnYK4wcdm5UaWI3fLcELopqLrHgLqNsdpHauzlQktfkHL3npOSwtj1Uz9oKBAzs7lFtVkV8j83voAz2D8fhw==}
+ dependencies:
+ '@types/history': 4.7.11
+ '@types/react': 18.2.6
+ '@types/react-router': 5.1.20
+ dev: true
+
+ /@types/react-router@5.1.20:
+ resolution: {integrity: sha512-jGjmu/ZqS7FjSH6owMcD5qpq19+1RS9DeVRqfl1FeBMxTDQAGwlMWOcs52NDoXaNKyG3d1cYQFMs9rCrb88o9Q==}
+ dependencies:
+ '@types/history': 4.7.11
+ '@types/react': 18.2.6
+ dev: true
+
+ /@types/react-transition-group@4.4.6:
+ resolution: {integrity: sha512-VnCdSxfcm08KjsJVQcfBmhEQAPnLB8G08hAxn39azX1qYBQ/5RVQuoHuKIcfKOdncuaUvEpFKFzEvbtIMsfVew==}
+ dependencies:
+ '@types/react': 18.2.6
+ dev: false
+
+ /@types/react@18.2.6:
+ resolution: {integrity: sha512-wRZClXn//zxCFW+ye/D2qY65UsYP1Fpex2YXorHc8awoNamkMZSvBxwxdYVInsHOZZd2Ppq8isnSzJL5Mpf8OA==}
+ dependencies:
+ '@types/prop-types': 15.7.5
+ '@types/scheduler': 0.16.3
+ csstype: 3.1.2
+
+ /@types/scheduler@0.16.3:
+ resolution: {integrity: sha512-5cJ8CB4yAx7BH1oMvdU0Jh9lrEXyPkar6F9G/ERswkCuvP4KQZfZkSjcMbAICCpQTN4OuZn8tz0HiKv9TGZgrQ==}
+
+ /@types/stack-utils@2.0.1:
+ resolution: {integrity: sha512-Hl219/BT5fLAaz6NDkSuhzasy49dwQS/DSdu4MdggFB8zcXv7vflBI3xp7FEmkmdDkBUI2bPUNeMttp2knYdxw==}
+ dev: true
+
+ /@types/testing-library__jest-dom@5.14.5:
+ resolution: {integrity: sha512-SBwbxYoyPIvxHbeHxTZX2Pe/74F/tX2/D3mMvzabdeJ25bBojfW0TyB8BHrbq/9zaaKICJZjLP+8r6AeZMFCuQ==}
+ dependencies:
+ '@types/jest': 29.5.1
+ dev: true
+
+ /@types/throttle-debounce@2.1.0:
+ resolution: {integrity: sha512-5eQEtSCoESnh2FsiLTxE121IiE60hnMqcb435fShf4bpLRjEu1Eoekht23y6zXS9Ts3l+Szu3TARnTsA0GkOkQ==}
+ dev: false
+
+ /@types/yargs-parser@21.0.0:
+ resolution: {integrity: sha512-iO9ZQHkZxHn4mSakYV0vFHAVDyEOIJQrV2uZ06HxEPcx+mt8swXoZHIbaaJ2crJYFfErySgktuTZ3BeLz+XmFA==}
+ dev: true
+
+ /@types/yargs@17.0.24:
+ resolution: {integrity: sha512-6i0aC7jV6QzQB8ne1joVZ0eSFIstHsCrobmOtghM11yGlH0j43FKL2UhWdELkyps0zuf7qVTUVCCR+tgSlyLLw==}
+ dependencies:
+ '@types/yargs-parser': 21.0.0
+ dev: true
+
+ /@vitejs/plugin-react@4.0.0(vite@4.3.8):
+ resolution: {integrity: sha512-HX0XzMjL3hhOYm+0s95pb0Z7F8O81G7joUHgfDd/9J/ZZf5k4xX6QAMFkKsHFxaHlf6X7GD7+XuaZ66ULiJuhQ==}
+ engines: {node: ^14.18.0 || >=16.0.0}
+ peerDependencies:
+ vite: ^4.2.0
+ dependencies:
+ '@babel/core': 7.21.8
+ '@babel/plugin-transform-react-jsx-self': 7.21.0(@babel/core@7.21.8)
+ '@babel/plugin-transform-react-jsx-source': 7.19.6(@babel/core@7.21.8)
+ react-refresh: 0.14.0
+ vite: 4.3.8
+ transitivePeerDependencies:
+ - supports-color
+ dev: true
+
+ /acorn-jsx@5.3.2(acorn@8.8.2):
+ resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==}
+ peerDependencies:
+ acorn: ^6.0.0 || ^7.0.0 || ^8.0.0
+ dependencies:
+ acorn: 8.8.2
+ dev: true
+
+ /acorn@8.8.2:
+ resolution: {integrity: sha512-xjIYgE8HBrkpd/sJqOGNspf8uHG+NOHGOw6a/Urj8taM2EXfdNAH2oFcPeIFfsv3+kz/mJrS5VuMqbNLjCa2vw==}
+ engines: {node: '>=0.4.0'}
+ hasBin: true
+ dev: true
+
+ /ajv@6.12.6:
+ resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==}
+ dependencies:
+ fast-deep-equal: 3.1.3
+ fast-json-stable-stringify: 2.1.0
+ json-schema-traverse: 0.4.1
+ uri-js: 4.4.1
+ dev: true
+
+ /ansi-regex@5.0.1:
+ resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /ansi-styles@3.2.1:
+ resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==}
+ engines: {node: '>=4'}
+ dependencies:
+ color-convert: 1.9.3
+
+ /ansi-styles@4.3.0:
+ resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==}
+ engines: {node: '>=8'}
+ dependencies:
+ color-convert: 2.0.1
+ dev: true
+
+ /ansi-styles@5.2.0:
+ resolution: {integrity: sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==}
+ engines: {node: '>=10'}
+ dev: true
+
+ /argparse@2.0.1:
+ resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==}
+
+ /aria-query@5.1.3:
+ resolution: {integrity: sha512-R5iJ5lkuHybztUfuOAznmboyjWq8O6sqNqtK7CLOqdydi54VNbORp49mb14KbWgG1QD3JFO9hJdZ+y4KutfdOQ==}
+ dependencies:
+ deep-equal: 2.2.1
+ dev: true
+
+ /array-buffer-byte-length@1.0.0:
+ resolution: {integrity: sha512-LPuwb2P+NrQw3XhxGc36+XSvuBPopovXYTR9Ew++Du9Yb/bx5AzBfrIsBoj0EZUifjQU+sHL21sseZ3jerWO/A==}
+ dependencies:
+ call-bind: 1.0.2
+ is-array-buffer: 3.0.2
+
+ /array-includes@3.1.6:
+ resolution: {integrity: sha512-sgTbLvL6cNnw24FnbaDyjmvddQ2ML8arZsgaJhoABMoplz/4QRhtrYS+alr1BUM1Bwp6dhx8vVCBSLG+StwOFw==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ get-intrinsic: 1.2.1
+ is-string: 1.0.7
+ dev: true
+
+ /array.prototype.flat@1.3.1:
+ resolution: {integrity: sha512-roTU0KWIOmJ4DRLmwKd19Otg0/mT3qPNt0Qb3GWW8iObuZXxrjB/pzn0R3hqpRSWg4HCwqx+0vwOnWnvlOyeIA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ es-shim-unscopables: 1.0.0
+ dev: true
+
+ /array.prototype.flatmap@1.3.1:
+ resolution: {integrity: sha512-8UGn9O1FDVvMNB0UlLv4voxRMze7+FpHyF5mSMRjWHUMlpoDViniy05870VlxhfgTnLbpuwTzvD76MTtWxB/mQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ es-shim-unscopables: 1.0.0
+ dev: true
+
+ /array.prototype.foreach@1.0.4:
+ resolution: {integrity: sha512-OYqqGR/56CopyheXNwdlJvFtbSvf2Z9RGvL20X6GvAuKePJ76L/D46BqZn3bITd36QA2Ti7Iy0UwVJaD/YwXZA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ es-array-method-boxes-properly: 1.0.0
+ get-intrinsic: 1.2.1
+ is-string: 1.0.7
+ dev: false
+
+ /array.prototype.tosorted@1.1.1:
+ resolution: {integrity: sha512-pZYPXPRl2PqWcsUs6LOMn+1f1532nEoPTYowBtqLwAW+W8vSVhkIGnmOX1t/UQjD6YGI0vcD2B1U7ZFGQH9jnQ==}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ es-shim-unscopables: 1.0.0
+ get-intrinsic: 1.2.1
+ dev: true
+
+ /ast-types-flow@0.0.7:
+ resolution: {integrity: sha512-eBvWn1lvIApYMhzQMsu9ciLfkBY499mFZlNqG+/9WR7PVlroQw0vG30cOQQbaKz3sCEc44TAOu2ykzqXSNnwag==}
+ dev: true
+
+ /async@3.2.4:
+ resolution: {integrity: sha512-iAB+JbDEGXhyIUavoDl9WP/Jj106Kz9DEn1DPgYw5ruDn0e3Wgi3sKFm55sASdGBNOQB8F59d9qQ7deqrHA8wQ==}
+ dev: true
+
+ /asynckit@0.4.0:
+ resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==}
+ dev: false
+
+ /attr-accept@2.2.2:
+ resolution: {integrity: sha512-7prDjvt9HmqiZ0cl5CRjtS84sEyhsHP2coDkaZKRKVfCDo9s7iw7ChVmar78Gu9pC4SoR/28wFu/G5JJhTnqEg==}
+ engines: {node: '>=4'}
+ dev: false
+
+ /autosuggest-highlight@3.3.4:
+ resolution: {integrity: sha512-j6RETBD2xYnrVcoV1S5R4t3WxOlWZKyDQjkwnggDPSjF5L4jV98ZltBpvPvbkM1HtoSe5o+bNrTHyjPbieGeYA==}
+ dependencies:
+ remove-accents: 0.4.4
+ dev: false
+
+ /available-typed-arrays@1.0.5:
+ resolution: {integrity: sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==}
+ engines: {node: '>= 0.4'}
+
+ /axe-core@4.7.1:
+ resolution: {integrity: sha512-sCXXUhA+cljomZ3ZAwb8i1p3oOlkABzPy08ZDAoGcYuvtBPlQ1Ytde129ArXyHWDhfeewq7rlx9F+cUx2SSlkg==}
+ engines: {node: '>=4'}
+ dev: true
+
+ /axios@1.4.0:
+ resolution: {integrity: sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==}
+ dependencies:
+ follow-redirects: 1.15.2
+ form-data: 4.0.0
+ proxy-from-env: 1.1.0
+ transitivePeerDependencies:
+ - debug
+ dev: false
+
+ /axobject-query@3.1.1:
+ resolution: {integrity: sha512-goKlv8DZrK9hUh975fnHzhNIO4jUnFCfv/dszV5VwUGDFjI6vQ2VwoyjYjYNEbBE8AH87TduWP5uyDR1D+Iteg==}
+ dependencies:
+ deep-equal: 2.2.1
+ dev: true
+
+ /babel-merge@3.0.0(@babel/core@7.21.8):
+ resolution: {integrity: sha512-eBOBtHnzt9xvnjpYNI5HmaPp/b2vMveE5XggzqHnQeHJ8mFIBrBv6WZEVIj5jJ2uwTItkqKo9gWzEEcBxEq0yw==}
+ peerDependencies:
+ '@babel/core': ^7.0.0
+ dependencies:
+ '@babel/core': 7.21.8
+ deepmerge: 2.2.1
+ object.omit: 3.0.0
+ dev: false
+
+ /babel-plugin-macros@3.1.0:
+ resolution: {integrity: sha512-Cg7TFGpIr01vOQNODXOOaGz2NpCU5gl8x1qJFbb6hbZxR7XrcE2vtbAsTAbJ7/xwJtUuJEw8K8Zr/AE0LHlesg==}
+ engines: {node: '>=10', npm: '>=6'}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ cosmiconfig: 7.1.0
+ resolve: 1.22.2
+ dev: false
+
+ /balanced-match@1.0.2:
+ resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==}
+
+ /big-integer@1.6.51:
+ resolution: {integrity: sha512-GPEid2Y9QU1Exl1rpO9B2IPJGHPSupF5GnVIP0blYvNOMer2bTvSWs1jGOUg04hTmu67nmLsQ9TBo1puaotBHg==}
+ engines: {node: '>=0.6'}
+ dev: false
+
+ /brace-expansion@1.1.11:
+ resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==}
+ dependencies:
+ balanced-match: 1.0.2
+ concat-map: 0.0.1
+
+ /brace-expansion@2.0.1:
+ resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==}
+ dependencies:
+ balanced-match: 1.0.2
+ dev: true
+
+ /braces@3.0.2:
+ resolution: {integrity: sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==}
+ engines: {node: '>=8'}
+ dependencies:
+ fill-range: 7.0.1
+ dev: true
+
+ /broadcast-channel@3.7.0:
+ resolution: {integrity: sha512-cIAKJXAxGJceNZGTZSBzMxzyOn72cVgPnKx4dc6LRjQgbaJUQqhy5rzL3zbMxkMWsGKkv2hSFkPRMEXfoMZ2Mg==}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ detect-node: 2.1.0
+ js-sha3: 0.8.0
+ microseconds: 0.2.0
+ nano-time: 1.0.0
+ oblivious-set: 1.0.0
+ rimraf: 3.0.2
+ unload: 2.2.0
+ dev: false
+
+ /browserslist@4.21.5:
+ resolution: {integrity: sha512-tUkiguQGW7S3IhB7N+c2MV/HZPSCPAAiYBZXLsBhFB/PCy6ZKKsZrmBayHV9fdGV/ARIfJ14NkxKzRDjvp7L6w==}
+ engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7}
+ hasBin: true
+ dependencies:
+ caniuse-lite: 1.0.30001488
+ electron-to-chromium: 1.4.402
+ node-releases: 2.0.11
+ update-browserslist-db: 1.0.11(browserslist@4.21.5)
+
+ /btoa@1.2.1:
+ resolution: {integrity: sha512-SB4/MIGlsiVkMcHmT+pSmIPoNDoHg+7cMzmt3Uxt628MTz2487DKSqK/fuhFBrkuqrYv5UCEnACpF4dTFNKc/g==}
+ engines: {node: '>= 0.4.0'}
+ hasBin: true
+ dev: true
+
+ /call-bind@1.0.2:
+ resolution: {integrity: sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==}
+ dependencies:
+ function-bind: 1.1.1
+ get-intrinsic: 1.2.1
+
+ /callsites@3.1.0:
+ resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==}
+ engines: {node: '>=6'}
+
+ /caniuse-lite@1.0.30001488:
+ resolution: {integrity: sha512-NORIQuuL4xGpIy6iCCQGN4iFjlBXtfKWIenlUuyZJumLRIindLb7wXM+GO8erEhb7vXfcnf4BAg2PrSDN5TNLQ==}
+
+ /case-anything@2.1.11:
+ resolution: {integrity: sha512-uzKDXzdM/x914cepWPzElU3y50NRKYhjkO4ittOHLq+rF6M0AgRLF/+yPR1tvwLNAh8WHEPTfhuciZGPfX+oyg==}
+ engines: {node: '>=12.13'}
+ dev: false
+
+ /chalk@2.4.2:
+ resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==}
+ engines: {node: '>=4'}
+ dependencies:
+ ansi-styles: 3.2.1
+ escape-string-regexp: 1.0.5
+ supports-color: 5.5.0
+
+ /chalk@3.0.0:
+ resolution: {integrity: sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==}
+ engines: {node: '>=8'}
+ dependencies:
+ ansi-styles: 4.3.0
+ supports-color: 7.2.0
+ dev: true
+
+ /chalk@4.1.2:
+ resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==}
+ engines: {node: '>=10'}
+ dependencies:
+ ansi-styles: 4.3.0
+ supports-color: 7.2.0
+ dev: true
+
+ /ci-info@3.8.0:
+ resolution: {integrity: sha512-eXTggHWSooYhq49F2opQhuHWgzucfF2YgODK4e1566GQs5BIfP30B0oenwBJHfWxAs2fyPB1s7Mg949zLf61Yw==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /cliui@7.0.4:
+ resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==}
+ dependencies:
+ string-width: 4.2.3
+ strip-ansi: 6.0.1
+ wrap-ansi: 7.0.0
+ dev: true
+
+ /clsx@1.2.1:
+ resolution: {integrity: sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==}
+ engines: {node: '>=6'}
+ dev: false
+
+ /color-convert@1.9.3:
+ resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==}
+ dependencies:
+ color-name: 1.1.3
+
+ /color-convert@2.0.1:
+ resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==}
+ engines: {node: '>=7.0.0'}
+ dependencies:
+ color-name: 1.1.4
+ dev: true
+
+ /color-name@1.1.3:
+ resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==}
+
+ /color-name@1.1.4:
+ resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==}
+ dev: true
+
+ /combined-stream@1.0.8:
+ resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==}
+ engines: {node: '>= 0.8'}
+ dependencies:
+ delayed-stream: 1.0.0
+ dev: false
+
+ /concat-map@0.0.1:
+ resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==}
+
+ /confusing-browser-globals@1.0.11:
+ resolution: {integrity: sha512-JsPKdmh8ZkmnHxDk55FZ1TqVLvEQTvoByJZRN9jzI0UjxK/QgAmsphz7PGtqgPieQZ/CQcHWXCR7ATDNhGe+YA==}
+ dev: true
+
+ /convert-source-map@1.9.0:
+ resolution: {integrity: sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==}
+
+ /cosmiconfig@7.1.0:
+ resolution: {integrity: sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==}
+ engines: {node: '>=10'}
+ dependencies:
+ '@types/parse-json': 4.0.0
+ import-fresh: 3.3.0
+ parse-json: 5.2.0
+ path-type: 4.0.0
+ yaml: 1.10.2
+ dev: false
+
+ /crelt@1.0.6:
+ resolution: {integrity: sha512-VQ2MBenTq1fWZUH9DJNGti7kKv6EeAuYr3cLwxUWhIu1baTaXh4Ib5W2CqHVqib4/MqbYGJqiL3Zb8GJZr3l4g==}
+ dev: false
+
+ /cross-spawn@7.0.3:
+ resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==}
+ engines: {node: '>= 8'}
+ dependencies:
+ path-key: 3.1.1
+ shebang-command: 2.0.0
+ which: 2.0.2
+ dev: true
+
+ /css-mediaquery@0.1.2:
+ resolution: {integrity: sha512-COtn4EROW5dBGlE/4PiKnh6rZpAPxDeFLaEEwt4i10jpDMFt2EhQGS79QmmrO+iKCHv0PU/HrOWEhijFd1x99Q==}
+ dev: false
+
+ /css.escape@1.5.1:
+ resolution: {integrity: sha512-YUifsXXuknHlUsmlgyY0PKzgPOr7/FjCePfHNt0jxm83wHZi44VDMQ7/fGNkjY3/jV1MC+1CmZbaHzugyeRtpg==}
+ dev: true
+
+ /csstype@3.1.2:
+ resolution: {integrity: sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ==}
+
+ /damerau-levenshtein@1.0.8:
+ resolution: {integrity: sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA==}
+ dev: true
+
+ /dash-get@1.0.2:
+ resolution: {integrity: sha512-4FbVrHDwfOASx7uQVxeiCTo7ggSdYZbqs8lH+WU6ViypPlDbe9y6IP5VVUDQBv9DcnyaiPT5XT0UWHgJ64zLeQ==}
+ dev: false
+
+ /date-fns@2.30.0:
+ resolution: {integrity: sha512-fnULvOpxnC5/Vg3NCiWelDsLiUc9bRwAPs/+LfTLNvetFCtCTN+yQz15C/fs4AwX1R9K5GLtLfn8QW+dWisaAw==}
+ engines: {node: '>=0.11'}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ dev: false
+
+ /debug@3.2.7:
+ resolution: {integrity: sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==}
+ peerDependencies:
+ supports-color: '*'
+ peerDependenciesMeta:
+ supports-color:
+ optional: true
+ dependencies:
+ ms: 2.1.3
+ dev: true
+
+ /debug@4.3.4:
+ resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==}
+ engines: {node: '>=6.0'}
+ peerDependencies:
+ supports-color: '*'
+ peerDependenciesMeta:
+ supports-color:
+ optional: true
+ dependencies:
+ ms: 2.1.2
+
+ /decode-uri-component@0.2.2:
+ resolution: {integrity: sha512-FqUYQ+8o158GyGTrMFJms9qh3CqTKvAqgqsTnkLI8sKu0028orqBhxNMFkFen0zGyg6epACD32pjVk58ngIErQ==}
+ engines: {node: '>=0.10'}
+ dev: false
+
+ /decode-uri-component@0.4.1:
+ resolution: {integrity: sha512-+8VxcR21HhTy8nOt6jf20w0c9CADrw1O8d+VZ/YzzCt4bJ3uBjw+D1q2osAB8RnpwwaeYBxy0HyKQxD5JBMuuQ==}
+ engines: {node: '>=14.16'}
+ dev: false
+
+ /deep-equal@2.2.1:
+ resolution: {integrity: sha512-lKdkdV6EOGoVn65XaOsPdH4rMxTZOnmFyuIkMjM1i5HHCbfjC97dawgTAy0deYNfuqUqW+Q5VrVaQYtUpSd6yQ==}
+ 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
+ dev: true
+
+ /deep-is@0.1.4:
+ resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==}
+ dev: true
+
+ /deepmerge@2.2.1:
+ resolution: {integrity: sha512-R9hc1Xa/NOBi9WRVUWg19rl1UB7Tt4kuPd+thNJgFZoxXsTz7ncaPaeIm+40oSGuP33DfMb4sZt1QIGiJzC4EA==}
+ engines: {node: '>=0.10.0'}
+ dev: false
+
+ /deepmerge@4.3.1:
+ resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==}
+ engines: {node: '>=0.10.0'}
+ dev: false
+
+ /define-properties@1.2.0:
+ resolution: {integrity: sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-property-descriptors: 1.0.0
+ object-keys: 1.1.1
+
+ /delayed-stream@1.0.0:
+ resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==}
+ engines: {node: '>=0.4.0'}
+ dev: false
+
+ /detect-node@2.1.0:
+ resolution: {integrity: sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==}
+ dev: false
+
+ /diff-sequences@29.4.3:
+ resolution: {integrity: sha512-ofrBgwpPhCD85kMKtE9RYFFq6OC1A89oW2vvgWZNCwxrUpRUILopY7lsYyMDSjc8g6U6aiO0Qubg6r4Wgt5ZnA==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dev: true
+
+ /doctrine@2.1.0:
+ resolution: {integrity: sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==}
+ engines: {node: '>=0.10.0'}
+ dependencies:
+ esutils: 2.0.3
+ dev: true
+
+ /doctrine@3.0.0:
+ resolution: {integrity: sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==}
+ engines: {node: '>=6.0.0'}
+ dependencies:
+ esutils: 2.0.3
+ dev: true
+
+ /dom-accessibility-api@0.5.16:
+ resolution: {integrity: sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==}
+ dev: true
+
+ /dom-helpers@5.2.1:
+ resolution: {integrity: sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ csstype: 3.1.2
+ dev: false
+
+ /dompurify@2.4.5:
+ resolution: {integrity: sha512-jggCCd+8Iqp4Tsz0nIvpcb22InKEBrGz5dw3EQJMs8HPJDsKbFIO3STYtAvCfDx26Muevn1MHVI0XxjgFfmiSA==}
+ dev: false
+
+ /duplexer@0.1.2:
+ resolution: {integrity: sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==}
+ dev: true
+
+ /ejs@3.1.9:
+ resolution: {integrity: sha512-rC+QVNMJWv+MtPgkt0y+0rVEIdbtxVADApW9JXrUVlzHetgcyczP/E7DJmWJ4fJCZF2cPcBk0laWO9ZHMG3DmQ==}
+ engines: {node: '>=0.10.0'}
+ hasBin: true
+ dependencies:
+ jake: 10.8.6
+ dev: true
+
+ /electron-to-chromium@1.4.402:
+ resolution: {integrity: sha512-gWYvJSkohOiBE6ecVYXkrDgNaUjo47QEKK0kQzmWyhkH+yoYiG44bwuicTGNSIQRG3WDMsWVZJLRnJnLNkbWvA==}
+
+ /emoji-regex@8.0.0:
+ resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==}
+ dev: true
+
+ /emoji-regex@9.2.2:
+ resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==}
+ dev: true
+
+ /entities@3.0.1:
+ resolution: {integrity: sha512-WiyBqoomrwMdFG1e0kqvASYfnlb0lp8M5o5Fw2OFq1hNZxxcNk8Ik0Xm7LxzBhuidnZB/UtBqVCgUz3kBOP51Q==}
+ engines: {node: '>=0.12'}
+ dev: false
+
+ /error-ex@1.3.2:
+ resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==}
+ dependencies:
+ is-arrayish: 0.2.1
+ dev: false
+
+ /es-abstract@1.21.2:
+ resolution: {integrity: sha512-y/B5POM2iBnIxCiernH1G7rC9qQoM77lLIMQLuob0zhp8C56Po81+2Nj0WFKnd0pNReDTnkYryc+zhOzpEIROg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ array-buffer-byte-length: 1.0.0
+ 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-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-length: 1.0.4
+ unbox-primitive: 1.0.2
+ which-typed-array: 1.1.9
+
+ /es-array-method-boxes-properly@1.0.0:
+ resolution: {integrity: sha512-wd6JXUmyHmt8T5a2xreUwKcGPq6f1f+WwIJkijUqiGcJz1qqnZgP6XIK+QyIWU5lT7imeNxUll48bziG+TSYcA==}
+ dev: false
+
+ /es-get-iterator@1.1.3:
+ resolution: {integrity: sha512-sPZmqHBe6JIiTfN5q2pEi//TwxmAFHwj/XEuYjTuse78i8KxaqMTTzxPoFKuzRpDpTJ+0NAbpfenkmH2rePtuw==}
+ dependencies:
+ call-bind: 1.0.2
+ get-intrinsic: 1.2.1
+ 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
+ dev: true
+
+ /es-set-tostringtag@2.0.1:
+ resolution: {integrity: sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ get-intrinsic: 1.2.1
+ has: 1.0.3
+ has-tostringtag: 1.0.0
+
+ /es-shim-unscopables@1.0.0:
+ resolution: {integrity: sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w==}
+ dependencies:
+ has: 1.0.3
+ dev: true
+
+ /es-to-primitive@1.2.1:
+ resolution: {integrity: sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ is-callable: 1.2.7
+ is-date-object: 1.0.5
+ is-symbol: 1.0.4
+
+ /esbuild@0.17.19:
+ resolution: {integrity: sha512-XQ0jAPFkK/u3LcVRcvVHQcTIqD6E2H1fvZMA5dQPSOWb3suUbWbfbRf94pjc0bNzRYLfIrDRQXr7X+LHIm5oHw==}
+ engines: {node: '>=12'}
+ hasBin: true
+ requiresBuild: true
+ optionalDependencies:
+ '@esbuild/android-arm': 0.17.19
+ '@esbuild/android-arm64': 0.17.19
+ '@esbuild/android-x64': 0.17.19
+ '@esbuild/darwin-arm64': 0.17.19
+ '@esbuild/darwin-x64': 0.17.19
+ '@esbuild/freebsd-arm64': 0.17.19
+ '@esbuild/freebsd-x64': 0.17.19
+ '@esbuild/linux-arm': 0.17.19
+ '@esbuild/linux-arm64': 0.17.19
+ '@esbuild/linux-ia32': 0.17.19
+ '@esbuild/linux-loong64': 0.17.19
+ '@esbuild/linux-mips64el': 0.17.19
+ '@esbuild/linux-ppc64': 0.17.19
+ '@esbuild/linux-riscv64': 0.17.19
+ '@esbuild/linux-s390x': 0.17.19
+ '@esbuild/linux-x64': 0.17.19
+ '@esbuild/netbsd-x64': 0.17.19
+ '@esbuild/openbsd-x64': 0.17.19
+ '@esbuild/sunos-x64': 0.17.19
+ '@esbuild/win32-arm64': 0.17.19
+ '@esbuild/win32-ia32': 0.17.19
+ '@esbuild/win32-x64': 0.17.19
+ dev: true
+
+ /escalade@3.1.1:
+ resolution: {integrity: sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==}
+ engines: {node: '>=6'}
+
+ /escape-html@1.0.3:
+ resolution: {integrity: sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==}
+ dev: true
+
+ /escape-string-regexp@1.0.5:
+ resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==}
+ engines: {node: '>=0.8.0'}
+
+ /escape-string-regexp@2.0.0:
+ resolution: {integrity: sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /escape-string-regexp@4.0.0:
+ resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==}
+ engines: {node: '>=10'}
+
+ /eslint-config-airbnb-base@15.0.0(eslint-plugin-import@2.27.5)(eslint@8.41.0):
+ resolution: {integrity: sha512-xaX3z4ZZIcFLvh2oUNvcX5oEofXda7giYmuplVxoOg5A7EXJMrUyqRgR+mhDhPK8LZ4PttFOBvCYDbX3sUoUig==}
+ engines: {node: ^10.12.0 || >=12.0.0}
+ peerDependencies:
+ eslint: ^7.32.0 || ^8.2.0
+ eslint-plugin-import: ^2.25.2
+ dependencies:
+ confusing-browser-globals: 1.0.11
+ eslint: 8.41.0
+ eslint-plugin-import: 2.27.5(eslint@8.41.0)
+ object.assign: 4.1.4
+ object.entries: 1.1.6
+ semver: 6.3.0
+ dev: true
+
+ /eslint-config-airbnb@19.0.4(eslint-plugin-import@2.27.5)(eslint-plugin-jsx-a11y@6.7.1)(eslint-plugin-react-hooks@4.6.0)(eslint-plugin-react@7.32.2)(eslint@8.41.0):
+ resolution: {integrity: sha512-T75QYQVQX57jiNgpF9r1KegMICE94VYwoFQyMGhrvc+lB8YF2E/M/PYDaQe1AJcWaEgqLE+ErXV1Og/+6Vyzew==}
+ engines: {node: ^10.12.0 || ^12.22.0 || ^14.17.0 || >=16.0.0}
+ peerDependencies:
+ eslint: ^7.32.0 || ^8.2.0
+ eslint-plugin-import: ^2.25.3
+ eslint-plugin-jsx-a11y: ^6.5.1
+ eslint-plugin-react: ^7.28.0
+ eslint-plugin-react-hooks: ^4.3.0
+ dependencies:
+ eslint: 8.41.0
+ eslint-config-airbnb-base: 15.0.0(eslint-plugin-import@2.27.5)(eslint@8.41.0)
+ eslint-plugin-import: 2.27.5(eslint@8.41.0)
+ eslint-plugin-jsx-a11y: 6.7.1(eslint@8.41.0)
+ eslint-plugin-react: 7.32.2(eslint@8.41.0)
+ eslint-plugin-react-hooks: 4.6.0(eslint@8.41.0)
+ object.assign: 4.1.4
+ object.entries: 1.1.6
+ dev: true
+
+ /eslint-import-resolver-node@0.3.7:
+ resolution: {integrity: sha512-gozW2blMLJCeFpBwugLTGyvVjNoeo1knonXAcatC6bjPBZitotxdWf7Gimr25N4c0AAOo4eOUfaG82IJPDpqCA==}
+ dependencies:
+ debug: 3.2.7
+ is-core-module: 2.12.1
+ resolve: 1.22.2
+ transitivePeerDependencies:
+ - supports-color
+ dev: true
+
+ /eslint-module-utils@2.8.0(eslint-import-resolver-node@0.3.7)(eslint@8.41.0):
+ resolution: {integrity: sha512-aWajIYfsqCKRDgUfjEXNN/JlrzauMuSEy5sbd7WXbtW3EH6A6MpwEh42c7qD+MqQo9QMJ6fWLAeIJynx0g6OAw==}
+ engines: {node: '>=4'}
+ peerDependencies:
+ '@typescript-eslint/parser': '*'
+ eslint: '*'
+ eslint-import-resolver-node: '*'
+ eslint-import-resolver-typescript: '*'
+ eslint-import-resolver-webpack: '*'
+ peerDependenciesMeta:
+ '@typescript-eslint/parser':
+ optional: true
+ eslint:
+ optional: true
+ eslint-import-resolver-node:
+ optional: true
+ eslint-import-resolver-typescript:
+ optional: true
+ eslint-import-resolver-webpack:
+ optional: true
+ dependencies:
+ debug: 3.2.7
+ eslint: 8.41.0
+ eslint-import-resolver-node: 0.3.7
+ transitivePeerDependencies:
+ - supports-color
+ dev: true
+
+ /eslint-plugin-import@2.27.5(eslint@8.41.0):
+ resolution: {integrity: sha512-LmEt3GVofgiGuiE+ORpnvP+kAm3h6MLZJ4Q5HCyHADofsb4VzXFsRiWj3c0OFiV+3DWFh0qg3v9gcPlfc3zRow==}
+ engines: {node: '>=4'}
+ peerDependencies:
+ '@typescript-eslint/parser': '*'
+ eslint: ^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8
+ peerDependenciesMeta:
+ '@typescript-eslint/parser':
+ optional: true
+ dependencies:
+ array-includes: 3.1.6
+ array.prototype.flat: 1.3.1
+ array.prototype.flatmap: 1.3.1
+ debug: 3.2.7
+ doctrine: 2.1.0
+ eslint: 8.41.0
+ eslint-import-resolver-node: 0.3.7
+ eslint-module-utils: 2.8.0(eslint-import-resolver-node@0.3.7)(eslint@8.41.0)
+ has: 1.0.3
+ is-core-module: 2.12.1
+ is-glob: 4.0.3
+ minimatch: 3.1.2
+ object.values: 1.1.6
+ resolve: 1.22.2
+ semver: 6.3.0
+ tsconfig-paths: 3.14.2
+ transitivePeerDependencies:
+ - eslint-import-resolver-typescript
+ - eslint-import-resolver-webpack
+ - supports-color
+ dev: true
+
+ /eslint-plugin-jsx-a11y@6.7.1(eslint@8.41.0):
+ resolution: {integrity: sha512-63Bog4iIethyo8smBklORknVjB0T2dwB8Mr/hIC+fBS0uyHdYYpzM/Ed+YC8VxTjlXHEWFOdmgwcDn1U2L9VCA==}
+ engines: {node: '>=4.0'}
+ peerDependencies:
+ eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8
+ dependencies:
+ '@babel/runtime': 7.21.5
+ 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.7.1
+ axobject-query: 3.1.1
+ damerau-levenshtein: 1.0.8
+ emoji-regex: 9.2.2
+ eslint: 8.41.0
+ 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
+ dev: true
+
+ /eslint-plugin-react-hooks@4.6.0(eslint@8.41.0):
+ resolution: {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
+ dependencies:
+ eslint: 8.41.0
+ dev: true
+
+ /eslint-plugin-react@7.32.2(eslint@8.41.0):
+ resolution: {integrity: sha512-t2fBMa+XzonrrNkyVirzKlvn5RXzzPwRHtMvLAtVZrt8oxgnTQaYbU6SXTOO1mwQgp1y5+toMSKInnzGr0Knqg==}
+ engines: {node: '>=4'}
+ peerDependencies:
+ eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8
+ dependencies:
+ array-includes: 3.1.6
+ array.prototype.flatmap: 1.3.1
+ array.prototype.tosorted: 1.1.1
+ doctrine: 2.1.0
+ eslint: 8.41.0
+ estraverse: 5.3.0
+ jsx-ast-utils: 3.3.3
+ 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.0
+ string.prototype.matchall: 4.0.8
+ dev: true
+
+ /eslint-scope@7.2.0:
+ resolution: {integrity: sha512-DYj5deGlHBfMt15J7rdtyKNq/Nqlv5KfU4iodrQ019XESsRnwXH9KAE0y3cwtUHDo2ob7CypAnCqefh6vioWRw==}
+ engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
+ dependencies:
+ esrecurse: 4.3.0
+ estraverse: 5.3.0
+ dev: true
+
+ /eslint-visitor-keys@3.4.1:
+ resolution: {integrity: sha512-pZnmmLwYzf+kWaM/Qgrvpen51upAktaaiI01nsJD/Yr3lMOdNtq0cxkrrg16w64VtisN6okbs7Q8AfGqj4c9fA==}
+ engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
+ dev: true
+
+ /eslint@8.41.0:
+ resolution: {integrity: sha512-WQDQpzGBOP5IrXPo4Hc0814r4/v2rrIsB0rhT7jtunIalgg6gYXWhRMOejVO8yH21T/FGaxjmFjBMNqcIlmH1Q==}
+ engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
+ hasBin: true
+ dependencies:
+ '@eslint-community/eslint-utils': 4.4.0(eslint@8.41.0)
+ '@eslint-community/regexpp': 4.5.1
+ '@eslint/eslintrc': 2.0.3
+ '@eslint/js': 8.41.0
+ '@humanwhocodes/config-array': 0.11.8
+ '@humanwhocodes/module-importer': 1.0.1
+ '@nodelib/fs.walk': 1.2.8
+ ajv: 6.12.6
+ chalk: 4.1.2
+ cross-spawn: 7.0.3
+ debug: 4.3.4
+ doctrine: 3.0.0
+ escape-string-regexp: 4.0.0
+ eslint-scope: 7.2.0
+ eslint-visitor-keys: 3.4.1
+ espree: 9.5.2
+ esquery: 1.5.0
+ esutils: 2.0.3
+ fast-deep-equal: 3.1.3
+ file-entry-cache: 6.0.1
+ find-up: 5.0.0
+ glob-parent: 6.0.2
+ globals: 13.20.0
+ graphemer: 1.4.0
+ ignore: 5.2.4
+ import-fresh: 3.3.0
+ imurmurhash: 0.1.4
+ is-glob: 4.0.3
+ 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.1
+ strip-ansi: 6.0.1
+ strip-json-comments: 3.1.1
+ text-table: 0.2.0
+ transitivePeerDependencies:
+ - supports-color
+ dev: true
+
+ /espree@9.5.2:
+ resolution: {integrity: sha512-7OASN1Wma5fum5SrNhFMAMJxOUAbhyfQ8dQ//PJaJbNw0URTPWqIghHWt1MmAANKhHZIYOHruW4Kw4ruUWOdGw==}
+ engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
+ dependencies:
+ acorn: 8.8.2
+ acorn-jsx: 5.3.2(acorn@8.8.2)
+ eslint-visitor-keys: 3.4.1
+ dev: true
+
+ /esquery@1.5.0:
+ resolution: {integrity: sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==}
+ engines: {node: '>=0.10'}
+ dependencies:
+ estraverse: 5.3.0
+ dev: true
+
+ /esrecurse@4.3.0:
+ resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==}
+ engines: {node: '>=4.0'}
+ dependencies:
+ estraverse: 5.3.0
+ dev: true
+
+ /estraverse@5.3.0:
+ resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==}
+ engines: {node: '>=4.0'}
+ dev: true
+
+ /esutils@2.0.3:
+ resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==}
+ engines: {node: '>=0.10.0'}
+ dev: true
+
+ /eventemitter3@4.0.7:
+ resolution: {integrity: sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==}
+ dev: false
+
+ /expect@29.5.0:
+ resolution: {integrity: sha512-yM7xqUrCO2JdpFo4XpM82t+PJBFybdqoQuJLDGeDX2ij8NZzqRHyu3Hp188/JX7SWqud+7t4MUdvcgGBICMHZg==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ '@jest/expect-utils': 29.5.0
+ jest-get-type: 29.4.3
+ jest-matcher-utils: 29.5.0
+ jest-message-util: 29.5.0
+ jest-util: 29.5.0
+ dev: true
+
+ /fast-deep-equal@3.1.3:
+ resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==}
+
+ /fast-json-stable-stringify@2.1.0:
+ resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==}
+ dev: true
+
+ /fast-levenshtein@2.0.6:
+ resolution: {integrity: sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==}
+ dev: true
+
+ /fastq@1.15.0:
+ resolution: {integrity: sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==}
+ dependencies:
+ reusify: 1.0.4
+ dev: true
+
+ /file-entry-cache@6.0.1:
+ resolution: {integrity: sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==}
+ engines: {node: ^10.12.0 || >=12.0.0}
+ dependencies:
+ flat-cache: 3.0.4
+ dev: true
+
+ /file-selector@0.5.0:
+ resolution: {integrity: sha512-s8KNnmIDTBoD0p9uJ9uD0XY38SCeBOtj0UMXyQSLg1Ypfrfj8+dAvwsLjYQkQ2GjhVtp2HrnF5cJzMhBjfD8HA==}
+ engines: {node: '>= 10'}
+ dependencies:
+ tslib: 2.5.2
+ dev: false
+
+ /filelist@1.0.4:
+ resolution: {integrity: sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==}
+ dependencies:
+ minimatch: 5.1.6
+ dev: true
+
+ /fill-range@7.0.1:
+ resolution: {integrity: sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==}
+ engines: {node: '>=8'}
+ dependencies:
+ to-regex-range: 5.0.1
+ dev: true
+
+ /filter-obj@1.1.0:
+ resolution: {integrity: sha512-8rXg1ZnX7xzy2NGDVkBVaAy+lSlPNwad13BtgSlLuxfIslyt5Vg64U7tFcCt4WS1R0hvtnQybT/IyCkGZ3DpXQ==}
+ engines: {node: '>=0.10.0'}
+ dev: false
+
+ /filter-obj@5.1.0:
+ resolution: {integrity: sha512-qWeTREPoT7I0bifpPUXtxkZJ1XJzxWtfoWWkdVGqa+eCr3SHW/Ocp89o8vLvbUuQnadybJpjOKu4V+RwO6sGng==}
+ engines: {node: '>=14.16'}
+ dev: false
+
+ /find-root@1.1.0:
+ resolution: {integrity: sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng==}
+ dev: false
+
+ /find-up@5.0.0:
+ resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==}
+ engines: {node: '>=10'}
+ dependencies:
+ locate-path: 6.0.0
+ path-exists: 4.0.0
+ dev: true
+
+ /flat-cache@3.0.4:
+ resolution: {integrity: sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==}
+ engines: {node: ^10.12.0 || >=12.0.0}
+ dependencies:
+ flatted: 3.2.7
+ rimraf: 3.0.2
+ dev: true
+
+ /flatted@3.2.7:
+ resolution: {integrity: sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==}
+ dev: true
+
+ /follow-redirects@1.15.2:
+ resolution: {integrity: sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==}
+ engines: {node: '>=4.0'}
+ peerDependencies:
+ debug: '*'
+ peerDependenciesMeta:
+ debug:
+ optional: true
+ dev: false
+
+ /for-each@0.3.3:
+ resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==}
+ dependencies:
+ is-callable: 1.2.7
+
+ /form-data@4.0.0:
+ resolution: {integrity: sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==}
+ engines: {node: '>= 6'}
+ dependencies:
+ asynckit: 0.4.0
+ combined-stream: 1.0.8
+ mime-types: 2.1.35
+ dev: false
+
+ /fs.realpath@1.0.0:
+ resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==}
+
+ /fsevents@2.3.2:
+ resolution: {integrity: sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==}
+ engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0}
+ os: [darwin]
+ requiresBuild: true
+ dev: true
+ optional: true
+
+ /function-bind@1.1.1:
+ resolution: {integrity: sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==}
+
+ /function.prototype.name@1.1.5:
+ resolution: {integrity: sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ functions-have-names: 1.2.3
+
+ /functions-have-names@1.2.3:
+ resolution: {integrity: sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==}
+
+ /gensync@1.0.0-beta.2:
+ resolution: {integrity: sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==}
+ engines: {node: '>=6.9.0'}
+
+ /get-caller-file@2.0.5:
+ resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==}
+ engines: {node: 6.* || 8.* || >= 10.*}
+ dev: true
+
+ /get-intrinsic@1.2.1:
+ resolution: {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
+
+ /get-symbol-description@1.0.0:
+ resolution: {integrity: sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ get-intrinsic: 1.2.1
+
+ /glob-parent@6.0.2:
+ resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==}
+ engines: {node: '>=10.13.0'}
+ dependencies:
+ is-glob: 4.0.3
+ dev: true
+
+ /glob@7.2.3:
+ resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==}
+ dependencies:
+ fs.realpath: 1.0.0
+ inflight: 1.0.6
+ inherits: 2.0.4
+ minimatch: 3.1.2
+ once: 1.4.0
+ path-is-absolute: 1.0.1
+
+ /globals@11.12.0:
+ resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==}
+ engines: {node: '>=4'}
+
+ /globals@13.20.0:
+ resolution: {integrity: sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ==}
+ engines: {node: '>=8'}
+ dependencies:
+ type-fest: 0.20.2
+ dev: true
+
+ /globalthis@1.0.3:
+ resolution: {integrity: sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ define-properties: 1.2.0
+
+ /gopd@1.0.1:
+ resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==}
+ dependencies:
+ get-intrinsic: 1.2.1
+
+ /graceful-fs@4.2.11:
+ resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==}
+ dev: true
+
+ /graphemer@1.4.0:
+ resolution: {integrity: sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==}
+ dev: true
+
+ /gzip-size@6.0.0:
+ resolution: {integrity: sha512-ax7ZYomf6jqPTQ4+XCpUGyXKHk5WweS+e05MBO4/y3WJ5RkmPXNKvX+bx1behVILVwr6JSQvZAku021CHPXG3Q==}
+ engines: {node: '>=10'}
+ dependencies:
+ duplexer: 0.1.2
+ dev: true
+
+ /has-bigints@1.0.2:
+ resolution: {integrity: sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==}
+
+ /has-flag@3.0.0:
+ resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==}
+ engines: {node: '>=4'}
+
+ /has-flag@4.0.0:
+ resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /has-property-descriptors@1.0.0:
+ resolution: {integrity: sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==}
+ dependencies:
+ get-intrinsic: 1.2.1
+
+ /has-proto@1.0.1:
+ resolution: {integrity: sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==}
+ engines: {node: '>= 0.4'}
+
+ /has-symbols@1.0.3:
+ resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==}
+ engines: {node: '>= 0.4'}
+
+ /has-tostringtag@1.0.0:
+ resolution: {integrity: sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-symbols: 1.0.3
+
+ /has@1.0.3:
+ resolution: {integrity: sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==}
+ engines: {node: '>= 0.4.0'}
+ dependencies:
+ function-bind: 1.1.1
+
+ /history@5.3.0:
+ resolution: {integrity: sha512-ZqaKwjjrAYUYfLG+htGaIIZ4nioX2L70ZUMIFysS3xvBsSG4x/n1V6TXV3N8ZYNuFGlDirFg32T7B6WOUPDYcQ==}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ dev: false
+
+ /hoist-non-react-statics@3.3.2:
+ resolution: {integrity: sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==}
+ dependencies:
+ react-is: 16.13.1
+ dev: false
+
+ /ignore@5.2.4:
+ resolution: {integrity: sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==}
+ engines: {node: '>= 4'}
+ dev: true
+
+ /import-fresh@3.3.0:
+ resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==}
+ engines: {node: '>=6'}
+ dependencies:
+ parent-module: 1.0.1
+ resolve-from: 4.0.0
+
+ /imurmurhash@0.1.4:
+ resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==}
+ engines: {node: '>=0.8.19'}
+ dev: true
+
+ /indent-string@4.0.0:
+ resolution: {integrity: sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /inflection@1.12.0:
+ resolution: {integrity: sha512-lRy4DxuIFWXlJU7ed8UiTJOSTqStqYdEb4CEbtXfNbkdj3nH1L+reUWiE10VWcJS2yR7tge8Z74pJjtBjNwj0w==}
+ engines: {'0': node >= 0.4.0}
+ dev: false
+
+ /inflight@1.0.6:
+ resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==}
+ dependencies:
+ once: 1.4.0
+ wrappy: 1.0.2
+
+ /inherits@2.0.4:
+ resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==}
+
+ /internal-slot@1.0.5:
+ resolution: {integrity: sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ get-intrinsic: 1.2.1
+ has: 1.0.3
+ side-channel: 1.0.4
+
+ /is-arguments@1.1.1:
+ resolution: {integrity: sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ has-tostringtag: 1.0.0
+ dev: true
+
+ /is-array-buffer@3.0.2:
+ resolution: {integrity: sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==}
+ dependencies:
+ call-bind: 1.0.2
+ get-intrinsic: 1.2.1
+ is-typed-array: 1.1.10
+
+ /is-arrayish@0.2.1:
+ resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==}
+ dev: false
+
+ /is-bigint@1.0.4:
+ resolution: {integrity: sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==}
+ dependencies:
+ has-bigints: 1.0.2
+
+ /is-boolean-object@1.1.2:
+ resolution: {integrity: sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ has-tostringtag: 1.0.0
+
+ /is-callable@1.2.7:
+ resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==}
+ engines: {node: '>= 0.4'}
+
+ /is-core-module@2.12.1:
+ resolution: {integrity: sha512-Q4ZuBAe2FUsKtyQJoQHlvP8OvBERxO3jEmy1I7hcRXcJBGGHFh/aJBswbXuS9sgrDH2QUO8ilkwNPHvHMd8clg==}
+ dependencies:
+ has: 1.0.3
+
+ /is-date-object@1.0.5:
+ resolution: {integrity: sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-tostringtag: 1.0.0
+
+ /is-docker@2.2.1:
+ resolution: {integrity: sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==}
+ engines: {node: '>=8'}
+ hasBin: true
+ dev: true
+
+ /is-extendable@1.0.1:
+ resolution: {integrity: sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==}
+ engines: {node: '>=0.10.0'}
+ dependencies:
+ is-plain-object: 2.0.4
+ dev: false
+
+ /is-extglob@2.1.1:
+ resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==}
+ engines: {node: '>=0.10.0'}
+ dev: true
+
+ /is-fullwidth-code-point@3.0.0:
+ resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /is-glob@4.0.3:
+ resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==}
+ engines: {node: '>=0.10.0'}
+ dependencies:
+ is-extglob: 2.1.1
+ dev: true
+
+ /is-map@2.0.2:
+ resolution: {integrity: sha512-cOZFQQozTha1f4MxLFzlgKYPTyj26picdZTx82hbc/Xf4K/tZOOXSCkMvU4pKioRXGDLJRn0GM7Upe7kR721yg==}
+ dev: true
+
+ /is-negative-zero@2.0.2:
+ resolution: {integrity: sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==}
+ engines: {node: '>= 0.4'}
+
+ /is-number-object@1.0.7:
+ resolution: {integrity: sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-tostringtag: 1.0.0
+
+ /is-number@7.0.0:
+ resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==}
+ engines: {node: '>=0.12.0'}
+ dev: true
+
+ /is-path-inside@3.0.3:
+ resolution: {integrity: sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /is-plain-object@2.0.4:
+ resolution: {integrity: sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==}
+ engines: {node: '>=0.10.0'}
+ dependencies:
+ isobject: 3.0.1
+ dev: false
+
+ /is-regex@1.1.4:
+ resolution: {integrity: sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ has-tostringtag: 1.0.0
+
+ /is-set@2.0.2:
+ resolution: {integrity: sha512-+2cnTEZeY5z/iXGbLhPrOAaK/Mau5k5eXq9j14CpRTftq0pAJu2MwVRSZhyZWBzx3o6X795Lz6Bpb6R0GKf37g==}
+ dev: true
+
+ /is-shared-array-buffer@1.0.2:
+ resolution: {integrity: sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==}
+ dependencies:
+ call-bind: 1.0.2
+
+ /is-string@1.0.7:
+ resolution: {integrity: sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-tostringtag: 1.0.0
+
+ /is-symbol@1.0.4:
+ resolution: {integrity: sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-symbols: 1.0.3
+
+ /is-typed-array@1.1.10:
+ resolution: {integrity: sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==}
+ engines: {node: '>= 0.4'}
+ 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
+
+ /is-weakmap@2.0.1:
+ resolution: {integrity: sha512-NSBR4kH5oVj1Uwvv970ruUkCV7O1mzgVFO4/rev2cLRda9Tm9HrL70ZPut4rOHgY0FNrUu9BCbXA2sdQ+x0chA==}
+ dev: true
+
+ /is-weakref@1.0.2:
+ resolution: {integrity: sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==}
+ dependencies:
+ call-bind: 1.0.2
+
+ /is-weakset@2.0.2:
+ resolution: {integrity: sha512-t2yVvttHkQktwnNNmBQ98AhENLdPUTDTE21uPqAQ0ARwQfGeQKRVS0NNurH7bTf7RrvcVn1OOge45CnBeHCSmg==}
+ dependencies:
+ call-bind: 1.0.2
+ get-intrinsic: 1.2.1
+ dev: true
+
+ /is-wsl@2.2.0:
+ resolution: {integrity: sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==}
+ engines: {node: '>=8'}
+ dependencies:
+ is-docker: 2.2.1
+ dev: true
+
+ /isarray@2.0.5:
+ resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==}
+ dev: true
+
+ /isexe@2.0.0:
+ resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==}
+ dev: true
+
+ /isobject@3.0.1:
+ resolution: {integrity: sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==}
+ engines: {node: '>=0.10.0'}
+ dev: false
+
+ /jake@10.8.6:
+ resolution: {integrity: sha512-G43Ub9IYEFfu72sua6rzooi8V8Gz2lkfk48rW20vEWCGizeaEPlKB1Kh8JIA84yQbiAEfqlPmSpGgCKKxH3rDA==}
+ engines: {node: '>=10'}
+ hasBin: true
+ dependencies:
+ async: 3.2.4
+ chalk: 4.1.2
+ filelist: 1.0.4
+ minimatch: 3.1.2
+ dev: true
+
+ /jest-diff@29.5.0:
+ resolution: {integrity: sha512-LtxijLLZBduXnHSniy0WMdaHjmQnt3g5sa16W4p0HqukYTTsyTW3GD1q41TyGl5YFXj/5B2U6dlh5FM1LIMgxw==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ chalk: 4.1.2
+ diff-sequences: 29.4.3
+ jest-get-type: 29.4.3
+ pretty-format: 29.5.0
+ dev: true
+
+ /jest-get-type@29.4.3:
+ resolution: {integrity: sha512-J5Xez4nRRMjk8emnTpWrlkyb9pfRQQanDrvWHhsR1+VUfbwxi30eVcZFlcdGInRibU4G5LwHXpI7IRHU0CY+gg==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dev: true
+
+ /jest-matcher-utils@29.5.0:
+ resolution: {integrity: sha512-lecRtgm/rjIK0CQ7LPQwzCs2VwW6WAahA55YBuI+xqmhm7LAaxokSB8C97yJeYyT+HvQkH741StzpU41wohhWw==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ chalk: 4.1.2
+ jest-diff: 29.5.0
+ jest-get-type: 29.4.3
+ pretty-format: 29.5.0
+ dev: true
+
+ /jest-message-util@29.5.0:
+ resolution: {integrity: sha512-Kijeg9Dag6CKtIDA7O21zNTACqD5MD/8HfIV8pdD94vFyFuer52SigdC3IQMhab3vACxXMiFk+yMHNdbqtyTGA==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ '@babel/code-frame': 7.21.4
+ '@jest/types': 29.5.0
+ '@types/stack-utils': 2.0.1
+ chalk: 4.1.2
+ graceful-fs: 4.2.11
+ micromatch: 4.0.5
+ pretty-format: 29.5.0
+ slash: 3.0.0
+ stack-utils: 2.0.6
+ dev: true
+
+ /jest-util@29.5.0:
+ resolution: {integrity: sha512-RYMgG/MTadOr5t8KdhejfvUU82MxsCu5MF6KuDUHl+NuwzUt+Sm6jJWxTJVrDR1j5M/gJVCPKQEpWXY+yIQ6lQ==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ '@jest/types': 29.5.0
+ '@types/node': 20.2.3
+ chalk: 4.1.2
+ ci-info: 3.8.0
+ graceful-fs: 4.2.11
+ picomatch: 2.3.1
+ dev: true
+
+ /js-sha3@0.8.0:
+ resolution: {integrity: sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==}
+ dev: false
+
+ /js-tokens@4.0.0:
+ resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==}
+
+ /js-yaml@4.1.0:
+ resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==}
+ hasBin: true
+ dependencies:
+ argparse: 2.0.1
+ dev: true
+
+ /jsesc@2.5.2:
+ resolution: {integrity: sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==}
+ engines: {node: '>=4'}
+ hasBin: true
+
+ /json-parse-even-better-errors@2.3.1:
+ resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==}
+ dev: false
+
+ /json-schema-traverse@0.4.1:
+ resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==}
+ dev: true
+
+ /json-stable-stringify-without-jsonify@1.0.1:
+ resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==}
+ dev: true
+
+ /json5@1.0.2:
+ resolution: {integrity: sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==}
+ hasBin: true
+ dependencies:
+ minimist: 1.2.8
+ dev: true
+
+ /json5@2.2.3:
+ resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==}
+ engines: {node: '>=6'}
+ hasBin: true
+
+ /jsonexport@3.2.0:
+ resolution: {integrity: sha512-GbO9ugb0YTZatPd/hqCGR0FSwbr82H6OzG04yzdrG7XOe4QZ0jhQ+kOsB29zqkzoYJLmLxbbrFiuwbQu891XnQ==}
+ hasBin: true
+ dev: false
+
+ /jsx-ast-utils@3.3.3:
+ resolution: {integrity: sha512-fYQHZTZ8jSfmWZ0iyzfwiU4WDX4HpHbMCZ3gPlWYiCl3BoeOTsqKBqnTVfH2rYT7eP5c3sVbeSPHnnJOaTrWiw==}
+ engines: {node: '>=4.0'}
+ dependencies:
+ array-includes: 3.1.6
+ object.assign: 4.1.4
+ dev: true
+
+ /language-subtag-registry@0.3.22:
+ resolution: {integrity: sha512-tN0MCzyWnoz/4nHS6uxdlFWoUZT7ABptwKPQ52Ea7URk6vll88bWBVhodtnlfEuCcKWNGoc+uGbw1cwa9IKh/w==}
+ dev: true
+
+ /language-tags@1.0.5:
+ resolution: {integrity: sha512-qJhlO9cGXi6hBGKoxEG/sKZDAHD5Hnu9Hs4WbOY3pCWXDhw0N8x1NenNzm2EnNLkLkk7J2SdxAkDSbb6ftT+UQ==}
+ dependencies:
+ language-subtag-registry: 0.3.22
+ dev: true
+
+ /levn@0.4.1:
+ resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==}
+ engines: {node: '>= 0.8.0'}
+ dependencies:
+ prelude-ls: 1.2.1
+ type-check: 0.4.0
+ dev: true
+
+ /lines-and-columns@1.2.4:
+ resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==}
+ dev: false
+
+ /linkify-it@4.0.1:
+ resolution: {integrity: sha512-C7bfi1UZmoj8+PQx22XyeXCuBlokoyWQL5pWSP+EI6nzRylyThouddufc2c1NDIcP9k5agmN9fLpA7VNJfIiqw==}
+ dependencies:
+ uc.micro: 1.0.6
+ dev: false
+
+ /linkifyjs@4.1.1:
+ resolution: {integrity: sha512-zFN/CTVmbcVef+WaDXT63dNzzkfRBKT1j464NJQkV7iSgJU0sLBus9W0HBwnXK13/hf168pbrx/V/bjEHOXNHA==}
+ dev: false
+
+ /locate-path@6.0.0:
+ resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==}
+ engines: {node: '>=10'}
+ dependencies:
+ p-locate: 5.0.0
+ dev: true
+
+ /lodash.merge@4.6.2:
+ resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==}
+ dev: true
+
+ /lodash@4.17.21:
+ resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==}
+
+ /loose-envify@1.4.0:
+ resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==}
+ hasBin: true
+ dependencies:
+ js-tokens: 4.0.0
+
+ /lru-cache@5.1.1:
+ resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==}
+ dependencies:
+ yallist: 3.1.1
+
+ /lz-string@1.5.0:
+ resolution: {integrity: sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==}
+ hasBin: true
+ dev: true
+
+ /make-error@1.3.6:
+ resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==}
+ dev: false
+
+ /markdown-it@13.0.1:
+ resolution: {integrity: sha512-lTlxriVoy2criHP0JKRhO2VDG9c2ypWCsT237eDiLqi09rmbKoUetyGHq2uOIRoRS//kfoJckS0eUzzkDR+k2Q==}
+ hasBin: true
+ dependencies:
+ argparse: 2.0.1
+ entities: 3.0.1
+ linkify-it: 4.0.1
+ mdurl: 1.0.1
+ uc.micro: 1.0.6
+ dev: false
+
+ /match-sorter@6.3.1:
+ resolution: {integrity: sha512-mxybbo3pPNuA+ZuCUhm5bwNkXrJTbsk5VWbR5wiwz/GC6LIiegBGn2w3O08UG/jdbYLinw51fSQ5xNU1U3MgBw==}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ remove-accents: 0.4.2
+ dev: false
+
+ /mdurl@1.0.1:
+ resolution: {integrity: sha512-/sKlQJCBYVY9Ers9hqzKou4H6V5UWc/M59TH2dvkt+84itfnq7uFOMLpOiOS4ujvHP4etln18fmIxA5R5fll0g==}
+ dev: false
+
+ /micromatch@4.0.5:
+ resolution: {integrity: sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==}
+ engines: {node: '>=8.6'}
+ dependencies:
+ braces: 3.0.2
+ picomatch: 2.3.1
+ dev: true
+
+ /microseconds@0.2.0:
+ resolution: {integrity: sha512-n7DHHMjR1avBbSpsTBj6fmMGh2AGrifVV4e+WYc3Q9lO+xnSZ3NyhcBND3vzzatt05LFhoKFRxrIyklmLlUtyA==}
+ dev: false
+
+ /mime-db@1.52.0:
+ resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==}
+ engines: {node: '>= 0.6'}
+ dev: false
+
+ /mime-types@2.1.35:
+ resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==}
+ engines: {node: '>= 0.6'}
+ dependencies:
+ mime-db: 1.52.0
+ dev: false
+
+ /min-indent@1.0.1:
+ resolution: {integrity: sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==}
+ engines: {node: '>=4'}
+ dev: true
+
+ /minimatch@3.1.2:
+ resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==}
+ dependencies:
+ brace-expansion: 1.1.11
+
+ /minimatch@5.1.6:
+ resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==}
+ engines: {node: '>=10'}
+ dependencies:
+ brace-expansion: 2.0.1
+ dev: true
+
+ /minimist@1.2.8:
+ resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==}
+ dev: true
+
+ /mkdirp@0.5.6:
+ resolution: {integrity: sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==}
+ hasBin: true
+ dependencies:
+ minimist: 1.2.8
+ dev: true
+
+ /ms@2.1.2:
+ resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==}
+
+ /ms@2.1.3:
+ resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==}
+ dev: true
+
+ /nano-time@1.0.0:
+ resolution: {integrity: sha512-flnngywOoQ0lLQOTRNexn2gGSNuM9bKj9RZAWSzhQ+UJYaAFG9bac4DW9VHjUAzrOaIcajHybCTHe/bkvozQqA==}
+ dependencies:
+ big-integer: 1.6.51
+ dev: false
+
+ /nanoid@3.3.6:
+ resolution: {integrity: sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA==}
+ engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1}
+ hasBin: true
+ dev: true
+
+ /natural-compare@1.4.0:
+ resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==}
+ dev: true
+
+ /node-polyglot@2.5.0:
+ resolution: {integrity: sha512-zXVwHNhFsG3mls+LKHxoHF70GQOL3FTDT3jH7ldkb95kG76RdU7F/NbvxV7D2hNIL9VpWXW6y78Fz+3KZkatRg==}
+ dependencies:
+ array.prototype.foreach: 1.0.4
+ has: 1.0.3
+ object.entries: 1.1.6
+ string.prototype.trim: 1.2.7
+ warning: 4.0.3
+ dev: false
+
+ /node-releases@2.0.11:
+ resolution: {integrity: sha512-+M0PwXeU80kRohZ3aT4J/OnR+l9/KD2nVLNNoRgFtnf+umQVFdGBAO2N8+nCnEi0xlh/Wk3zOGC+vNNx+uM79Q==}
+
+ /object-assign@4.1.1:
+ resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==}
+ engines: {node: '>=0.10.0'}
+
+ /object-inspect@1.12.3:
+ resolution: {integrity: sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==}
+
+ /object-is@1.1.5:
+ resolution: {integrity: sha512-3cyDsyHgtmi7I7DfSSI2LDp6SK2lwvtbg0p0R1e0RvTqF5ceGx+K2dfSjm1bKDMVCFEDAQvy+o8c6a7VujOddw==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ dev: true
+
+ /object-keys@1.1.1:
+ resolution: {integrity: sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==}
+ engines: {node: '>= 0.4'}
+
+ /object.assign@4.1.4:
+ resolution: {integrity: sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ has-symbols: 1.0.3
+ object-keys: 1.1.1
+
+ /object.entries@1.1.6:
+ resolution: {integrity: sha512-leTPzo4Zvg3pmbQ3rDK69Rl8GQvIqMWubrkxONG9/ojtFE2rD9fjMKfSI5BxW3osRH1m6VdzmqK8oAY9aT4x5w==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+
+ /object.fromentries@2.0.6:
+ resolution: {integrity: sha512-VciD13dswC4j1Xt5394WR4MzmAQmlgN72phd/riNp9vtD7tp4QQWJ0R4wvclXcafgcYK8veHRed2W6XeGBvcfg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ dev: true
+
+ /object.hasown@1.1.2:
+ resolution: {integrity: sha512-B5UIT3J1W+WuWIU55h0mjlwaqxiE5vYENJXIXZ4VFe05pNYrkKuK0U/6aFcb0pKywYJh7IhfoqUfKVmrJJHZHw==}
+ dependencies:
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ dev: true
+
+ /object.omit@3.0.0:
+ resolution: {integrity: sha512-EO+BCv6LJfu+gBIF3ggLicFebFLN5zqzz/WWJlMFfkMyGth+oBkhxzDl0wx2W4GkLzuQs/FsSkXZb2IMWQqmBQ==}
+ engines: {node: '>=0.10.0'}
+ dependencies:
+ is-extendable: 1.0.1
+ dev: false
+
+ /object.pick@1.3.0:
+ resolution: {integrity: sha512-tqa/UMy/CCoYmj+H5qc07qvSL9dqcs/WZENZ1JbtWBlATP+iVOe778gE6MSijnyCnORzDuX6hU+LA4SZ09YjFQ==}
+ engines: {node: '>=0.10.0'}
+ dependencies:
+ isobject: 3.0.1
+ dev: false
+
+ /object.values@1.1.6:
+ resolution: {integrity: sha512-FVVTkD1vENCsAcwNs9k6jea2uHC/X0+JcjG8YA60FN5CMaJmG95wT9jek/xX9nornqGRrBkKtzuAu2wuHpKqvw==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ dev: true
+
+ /oblivious-set@1.0.0:
+ resolution: {integrity: sha512-z+pI07qxo4c2CulUHCDf9lcqDlMSo72N/4rLUpRXf6fu+q8vjt8y0xS+Tlf8NTJDdTXHbdeO1n3MlbctwEoXZw==}
+ dev: false
+
+ /once@1.4.0:
+ resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==}
+ dependencies:
+ wrappy: 1.0.2
+
+ /open@7.4.2:
+ resolution: {integrity: sha512-MVHddDVweXZF3awtlAS+6pgKLlm/JgxZ90+/NBurBoQctVOOB/zDdVjcyPzQ+0laDGbsWgrRkflI65sQeOgT9Q==}
+ engines: {node: '>=8'}
+ dependencies:
+ is-docker: 2.2.1
+ is-wsl: 2.2.0
+ dev: true
+
+ /optionator@0.9.1:
+ resolution: {integrity: sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==}
+ engines: {node: '>= 0.8.0'}
+ dependencies:
+ deep-is: 0.1.4
+ fast-levenshtein: 2.0.6
+ levn: 0.4.1
+ prelude-ls: 1.2.1
+ type-check: 0.4.0
+ word-wrap: 1.2.3
+ dev: true
+
+ /orderedmap@2.1.1:
+ resolution: {integrity: sha512-TvAWxi0nDe1j/rtMcWcIj94+Ffe6n7zhow33h40SKxmsmozs6dz/e+EajymfoFcHd7sxNn8yHM8839uixMOV6g==}
+ dev: false
+
+ /p-limit@3.1.0:
+ resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==}
+ engines: {node: '>=10'}
+ dependencies:
+ yocto-queue: 0.1.0
+ dev: true
+
+ /p-locate@5.0.0:
+ resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==}
+ engines: {node: '>=10'}
+ dependencies:
+ p-limit: 3.1.0
+ dev: true
+
+ /parent-module@1.0.1:
+ resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==}
+ engines: {node: '>=6'}
+ dependencies:
+ callsites: 3.1.0
+
+ /parse-json@5.2.0:
+ resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==}
+ engines: {node: '>=8'}
+ dependencies:
+ '@babel/code-frame': 7.21.4
+ error-ex: 1.3.2
+ json-parse-even-better-errors: 2.3.1
+ lines-and-columns: 1.2.4
+ dev: false
+
+ /path-exists@4.0.0:
+ resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /path-is-absolute@1.0.1:
+ resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==}
+ engines: {node: '>=0.10.0'}
+
+ /path-key@3.1.1:
+ resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /path-parse@1.0.7:
+ resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==}
+
+ /path-type@4.0.0:
+ resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==}
+ engines: {node: '>=8'}
+ dev: false
+
+ /picocolors@1.0.0:
+ resolution: {integrity: sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==}
+
+ /picomatch@2.3.1:
+ resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==}
+ engines: {node: '>=8.6'}
+ dev: true
+
+ /postcss@8.4.23:
+ resolution: {integrity: sha512-bQ3qMcpF6A/YjR55xtoTr0jGOlnPOKAIMdOWiv0EIT6HVPEaJiJB4NLljSbiHoC2RX7DN5Uvjtpbg1NPdwv1oA==}
+ engines: {node: ^10 || ^12 || >=14}
+ dependencies:
+ nanoid: 3.3.6
+ picocolors: 1.0.0
+ source-map-js: 1.0.2
+ dev: true
+
+ /prelude-ls@1.2.1:
+ resolution: {integrity: sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==}
+ engines: {node: '>= 0.8.0'}
+ dev: true
+
+ /pretty-format@27.5.1:
+ resolution: {integrity: sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==}
+ engines: {node: ^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0}
+ dependencies:
+ ansi-regex: 5.0.1
+ ansi-styles: 5.2.0
+ react-is: 17.0.2
+ dev: true
+
+ /pretty-format@29.5.0:
+ resolution: {integrity: sha512-V2mGkI31qdttvTFX7Mt4efOqHXqJWMu4/r66Xh3Z3BwZaPfPJgp6/gbwoujRpPUtfEF6AUUWx3Jim3GCw5g/Qw==}
+ engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
+ dependencies:
+ '@jest/schemas': 29.4.3
+ ansi-styles: 5.2.0
+ react-is: 18.2.0
+ dev: true
+
+ /prop-types@15.8.1:
+ resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==}
+ dependencies:
+ loose-envify: 1.4.0
+ object-assign: 4.1.1
+ react-is: 16.13.1
+
+ /prosemirror-changeset@2.2.1:
+ resolution: {integrity: sha512-J7msc6wbxB4ekDFj+n9gTW/jav/p53kdlivvuppHsrZXCaQdVgRghoZbSS3kwrRyAstRVQ4/+u5k7YfLgkkQvQ==}
+ dependencies:
+ prosemirror-transform: 1.7.2
+ dev: false
+
+ /prosemirror-collab@1.3.1:
+ resolution: {integrity: sha512-4SnynYR9TTYaQVXd/ieUvsVV4PDMBzrq2xPUWutHivDuOshZXqQ5rGbZM84HEaXKbLdItse7weMGOUdDVcLKEQ==}
+ dependencies:
+ prosemirror-state: 1.4.3
+ dev: false
+
+ /prosemirror-commands@1.5.2:
+ resolution: {integrity: sha512-hgLcPaakxH8tu6YvVAaILV2tXYsW3rAdDR8WNkeKGcgeMVQg3/TMhPdVoh7iAmfgVjZGtcOSjKiQaoeKjzd2mQ==}
+ dependencies:
+ prosemirror-model: 1.19.1
+ prosemirror-state: 1.4.3
+ prosemirror-transform: 1.7.2
+ dev: false
+
+ /prosemirror-dropcursor@1.8.1:
+ resolution: {integrity: sha512-M30WJdJZLyXHi3N8vxN6Zh5O8ZBbQCz0gURTfPmTIBNQ5pxrdU7A58QkNqfa98YEjSAL1HUyyU34f6Pm5xBSGw==}
+ dependencies:
+ prosemirror-state: 1.4.3
+ prosemirror-transform: 1.7.2
+ prosemirror-view: 1.31.3
+ dev: false
+
+ /prosemirror-gapcursor@1.3.2:
+ resolution: {integrity: sha512-wtjswVBd2vaQRrnYZaBCbyDqr232Ed4p2QPtRIUK5FuqHYKGWkEwl08oQM4Tw7DOR0FsasARV5uJFvMZWxdNxQ==}
+ dependencies:
+ prosemirror-keymap: 1.2.2
+ prosemirror-model: 1.19.1
+ prosemirror-state: 1.4.3
+ prosemirror-view: 1.31.3
+ dev: false
+
+ /prosemirror-history@1.3.2:
+ resolution: {integrity: sha512-/zm0XoU/N/+u7i5zepjmZAEnpvjDtzoPWW6VmKptcAnPadN/SStsBjMImdCEbb3seiNTpveziPTIrXQbHLtU1g==}
+ dependencies:
+ prosemirror-state: 1.4.3
+ prosemirror-transform: 1.7.2
+ prosemirror-view: 1.31.3
+ rope-sequence: 1.3.4
+ dev: false
+
+ /prosemirror-inputrules@1.2.1:
+ resolution: {integrity: sha512-3LrWJX1+ULRh5SZvbIQlwZafOXqp1XuV21MGBu/i5xsztd+9VD15x6OtN6mdqSFI7/8Y77gYUbQ6vwwJ4mr6QQ==}
+ dependencies:
+ prosemirror-state: 1.4.3
+ prosemirror-transform: 1.7.2
+ dev: false
+
+ /prosemirror-keymap@1.2.2:
+ resolution: {integrity: sha512-EAlXoksqC6Vbocqc0GtzCruZEzYgrn+iiGnNjsJsH4mrnIGex4qbLdWWNza3AW5W36ZRrlBID0eM6bdKH4OStQ==}
+ dependencies:
+ prosemirror-state: 1.4.3
+ w3c-keyname: 2.2.7
+ dev: false
+
+ /prosemirror-markdown@1.11.0:
+ resolution: {integrity: sha512-yP9mZqPRstjZhhf3yykCQNE3AijxARrHe4e7esV9A+gp4cnGOH4QvrKYPpXLHspNWyvJJ+0URH+iIvV5qP1I2Q==}
+ dependencies:
+ markdown-it: 13.0.1
+ prosemirror-model: 1.19.1
+ dev: false
+
+ /prosemirror-menu@1.2.2:
+ resolution: {integrity: sha512-437HIWTq4F9cTX+kPfqZWWm+luJm95Aut/mLUy+9OMrOml0bmWDS26ceC6SNfb2/S94et1sZ186vLO7pDHzxSw==}
+ dependencies:
+ crelt: 1.0.6
+ prosemirror-commands: 1.5.2
+ prosemirror-history: 1.3.2
+ prosemirror-state: 1.4.3
+ dev: false
+
+ /prosemirror-model@1.19.1:
+ resolution: {integrity: sha512-RpV0fZfy74DEO9GPRbGcG6xN33KuqEvlLE2V0e5CXUGs3xkZsiJfx1dcYPU57+606NVYCaDN1riFXdXBQRaRcg==}
+ dependencies:
+ orderedmap: 2.1.1
+ dev: false
+
+ /prosemirror-schema-basic@1.2.2:
+ resolution: {integrity: sha512-/dT4JFEGyO7QnNTe9UaKUhjDXbTNkiWTq/N4VpKaF79bBjSExVV2NXmJpcM7z/gD7mbqNjxbmWW5nf1iNSSGnw==}
+ dependencies:
+ prosemirror-model: 1.19.1
+ dev: false
+
+ /prosemirror-schema-list@1.2.3:
+ resolution: {integrity: sha512-HD8yjDOusz7JB3oBFCaMOpEN9Z9DZttLr6tcASjnvKMc0qTyX5xgAN8YiMFFEcwyhF7WZrZ2YQkAwzsn8ICVbQ==}
+ dependencies:
+ prosemirror-model: 1.19.1
+ prosemirror-state: 1.4.3
+ prosemirror-transform: 1.7.2
+ dev: false
+
+ /prosemirror-state@1.4.3:
+ resolution: {integrity: sha512-goFKORVbvPuAQaXhpbemJFRKJ2aixr+AZMGiquiqKxaucC6hlpHNZHWgz5R7dS4roHiwq9vDctE//CZ++o0W1Q==}
+ dependencies:
+ prosemirror-model: 1.19.1
+ prosemirror-transform: 1.7.2
+ prosemirror-view: 1.31.3
+ dev: false
+
+ /prosemirror-tables@1.3.2:
+ resolution: {integrity: sha512-/9JTeN6s58Zq66HXaxP6uf8PAmc7XXKZFPlOGVtLvxEd6xBP6WtzaJB9wBjiGUzwbdhdMEy7V62yuHqk/3VrnQ==}
+ dependencies:
+ prosemirror-keymap: 1.2.2
+ prosemirror-model: 1.19.1
+ prosemirror-state: 1.4.3
+ prosemirror-transform: 1.7.2
+ prosemirror-view: 1.31.3
+ dev: false
+
+ /prosemirror-trailing-node@2.0.4(prosemirror-model@1.19.1)(prosemirror-state@1.4.3)(prosemirror-view@1.31.3):
+ resolution: {integrity: sha512-0Yl9w7IdHkaCdqR+NE3FOucePME4OmiGcybnF1iasarEILP5U8+4xTnl53yafULjmwcg1SrSG65Hg7Zk2H2v3g==}
+ peerDependencies:
+ prosemirror-model: ^1.19.0
+ prosemirror-state: ^1.4.2
+ prosemirror-view: ^1.30.2
+ dependencies:
+ '@babel/runtime': 7.21.5
+ '@remirror/core-constants': 2.0.1
+ '@remirror/core-helpers': 2.0.3
+ escape-string-regexp: 4.0.0
+ prosemirror-model: 1.19.1
+ prosemirror-state: 1.4.3
+ prosemirror-view: 1.31.3
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /prosemirror-transform@1.7.2:
+ resolution: {integrity: sha512-b94lVUdA9NyaYRb2WuGSgb5YANiITa05dtew9eSK+KkYu64BCnU27WhJPE95gAWAnhV57CM3FabWXM23gri8Kg==}
+ dependencies:
+ prosemirror-model: 1.19.1
+ dev: false
+
+ /prosemirror-view@1.31.3:
+ resolution: {integrity: sha512-UYDa8WxRFZm0xQLXiPJUVTl6H08Fn0IUVDootA7ZlQwzooqVWnBOXLovJyyTKgws1nprfsPhhlvWgt2jo4ZA6g==}
+ dependencies:
+ prosemirror-model: 1.19.1
+ prosemirror-state: 1.4.3
+ prosemirror-transform: 1.7.2
+ dev: false
+
+ /proxy-from-env@1.1.0:
+ resolution: {integrity: sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==}
+ dev: false
+
+ /punycode@2.3.0:
+ resolution: {integrity: sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==}
+ engines: {node: '>=6'}
+ dev: true
+
+ /query-string@7.1.3:
+ resolution: {integrity: sha512-hh2WYhq4fi8+b+/2Kg9CEge4fDPvHS534aOOvOZeQ3+Vf2mCFsaFBYj0i+iXcAq6I9Vzp5fjMFBlONvayDC1qg==}
+ engines: {node: '>=6'}
+ dependencies:
+ decode-uri-component: 0.2.2
+ filter-obj: 1.1.0
+ split-on-first: 1.1.0
+ strict-uri-encode: 2.0.0
+ dev: false
+
+ /query-string@8.1.0:
+ resolution: {integrity: sha512-BFQeWxJOZxZGix7y+SByG3F36dA0AbTy9o6pSmKFcFz7DAj0re9Frkty3saBn3nHo3D0oZJ/+rx3r8H8r8Jbpw==}
+ engines: {node: '>=14.16'}
+ dependencies:
+ decode-uri-component: 0.4.1
+ filter-obj: 5.1.0
+ split-on-first: 3.0.0
+ dev: false
+
+ /queue-microtask@1.2.3:
+ resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==}
+ dev: true
+
+ /ra-core@4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0):
+ resolution: {integrity: sha512-ai3ekdnkYmNWRP6m79zCs03+O3zBQsRxon/br5rsWodAtTa0j1oqjXNEVoQNipvXBmeCjnKENN9Gzc3wtUUZjA==}
+ peerDependencies:
+ history: ^5.1.0
+ react: ^16.9.0 || ^17.0.0 || ^18.0.0
+ react-dom: ^16.9.0 || ^17.0.0 || ^18.0.0
+ react-hook-form: ^7.40.0
+ react-router: ^6.1.0
+ react-router-dom: ^6.1.0
+ dependencies:
+ clsx: 1.2.1
+ date-fns: 2.30.0
+ eventemitter3: 4.0.7
+ history: 5.3.0
+ inflection: 1.12.0
+ jsonexport: 3.2.0
+ lodash: 4.17.21
+ prop-types: 15.8.1
+ query-string: 7.1.3
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ react-hook-form: 7.43.9(react@18.2.0)
+ react-is: 17.0.2
+ react-query: 3.39.3(react-dom@18.2.0)(react@18.2.0)
+ react-router: 6.11.2(react@18.2.0)
+ react-router-dom: 6.11.2(react-dom@18.2.0)(react@18.2.0)
+ transitivePeerDependencies:
+ - react-native
+ dev: false
+
+ /ra-data-simple-rest@4.10.3(ra-core@4.10.3):
+ resolution: {integrity: sha512-7Dq+5sNbdKfP8IZj56jP+cVhVo0J6gFtdIqWsN/YsxV1kEMDVHRp35VSlAJPLhAApvVhQ9Qoalvud3t8MaSGRg==}
+ peerDependencies:
+ ra-core: ^4.0.0
+ dependencies:
+ query-string: 7.1.3
+ ra-core: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ dev: false
+
+ /ra-i18n-polyglot@4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0):
+ resolution: {integrity: sha512-ZrDa8Z+AjMnflzTi7G2N6JYZS3tYh7vEf7qo8qpxGFtOG2ADGMg3p1N+UyqgxtCtm7GG+G+Alll0Ln5LwQBnuQ==}
+ dependencies:
+ node-polyglot: 2.5.0
+ ra-core: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ transitivePeerDependencies:
+ - history
+ - react
+ - react-dom
+ - react-hook-form
+ - react-native
+ - react-router
+ - react-router-dom
+ dev: false
+
+ /ra-input-rich-text@4.10.3(@mui/icons-material@5.11.16)(@mui/material@5.13.1)(ra-core@4.10.3)(ra-ui-materialui@4.10.3)(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-/Zi6Fi/4mtNRteiMQ3wEkrXgkgFH4GM62jQ1bIX1n6qJ74xUNpxpA5eC1fst7TB+EiSsfYcGGTOrVt+FpeVvGA==}
+ peerDependencies:
+ '@mui/icons-material': ^5.0.1
+ '@mui/material': ^5.0.2
+ ra-core: ^4.0.0
+ ra-ui-materialui: ^4.0.0
+ react: ^16.9.0 || ^17.0.0 || ^18.0.0
+ react-dom: ^16.9.0 || ^17.0.0 || ^18.0.0
+ dependencies:
+ '@mui/icons-material': 5.11.16(@mui/material@5.13.1)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/material': 5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0)
+ '@tiptap/core': 2.0.3(@tiptap/pm@2.0.3)
+ '@tiptap/extension-color': 2.0.3(@tiptap/core@2.0.3)(@tiptap/extension-text-style@2.0.3)
+ '@tiptap/extension-highlight': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-image': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-link': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-placeholder': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)
+ '@tiptap/extension-text-align': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-text-style': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/extension-underline': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/pm': 2.0.3(@tiptap/core@2.0.3)
+ '@tiptap/react': 2.0.3(@tiptap/core@2.0.3)(@tiptap/pm@2.0.3)(react-dom@18.2.0)(react@18.2.0)
+ '@tiptap/starter-kit': 2.0.3(@tiptap/pm@2.0.3)
+ clsx: 1.2.1
+ ra-core: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ ra-ui-materialui: 4.10.3(@mui/icons-material@5.11.16)(@mui/material@5.13.1)(ra-core@4.10.3)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ transitivePeerDependencies:
+ - supports-color
+ dev: false
+
+ /ra-language-english@4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0):
+ resolution: {integrity: sha512-IkAvfk5V/O32LAsMe/CrhGKIY2jPmDapvKFPvT2EkzhsMbi3J2uuTYHXdMX0f2M3MpJOJ+0ljssXIVfm35QbXw==}
+ dependencies:
+ ra-core: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ transitivePeerDependencies:
+ - history
+ - react
+ - react-dom
+ - react-hook-form
+ - react-native
+ - react-router
+ - react-router-dom
+ dev: false
+
+ /ra-ui-materialui@4.10.3(@mui/icons-material@5.11.16)(@mui/material@5.13.1)(ra-core@4.10.3)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0):
+ resolution: {integrity: sha512-GBEyD/5D9BT8+trJg4lMXNNJIjckVBtFQHsrYTa/oxb1uRIouAdO5lxNg9Jd/mZVKI/OQ5kQl5ruqh4B0qKWfw==}
+ peerDependencies:
+ '@mui/icons-material': ^5.0.1
+ '@mui/material': ^5.0.2
+ ra-core: ^4.0.0
+ react: ^16.9.0 || ^17.0.0 || ^18.0.0
+ react-dom: ^16.9.0 || ^17.0.0 || ^18.0.0
+ react-hook-form: '*'
+ react-router: ^6.1.0
+ react-router-dom: ^6.1.0
+ dependencies:
+ '@mui/icons-material': 5.11.16(@mui/material@5.13.1)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/material': 5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0)
+ autosuggest-highlight: 3.3.4
+ clsx: 1.2.1
+ css-mediaquery: 0.1.2
+ dompurify: 2.4.5
+ inflection: 1.12.0
+ jsonexport: 3.2.0
+ lodash: 4.17.21
+ prop-types: 15.8.1
+ query-string: 7.1.3
+ ra-core: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ react-dropzone: 12.1.0(react@18.2.0)
+ react-error-boundary: 3.1.4(react@18.2.0)
+ react-hook-form: 7.43.9(react@18.2.0)
+ react-query: 3.39.3(react-dom@18.2.0)(react@18.2.0)
+ react-router: 6.11.2(react@18.2.0)
+ react-router-dom: 6.11.2(react-dom@18.2.0)(react@18.2.0)
+ react-transition-group: 4.4.5(react-dom@18.2.0)(react@18.2.0)
+ transitivePeerDependencies:
+ - react-native
+ dev: false
+
+ /react-admin@4.10.3(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-ZeCOO0pnyZZbMFX3j60jz+Zz50g4n3xYmeOS2K4LKtEdKKEtxSTbz/JGHbtp4mqmwOX0NkuiUJxJYkBz6CA2Og==}
+ peerDependencies:
+ react: ^16.9.0 || ^17.0.0 || ^18.0.0
+ react-dom: ^16.9.0 || ^17.0.0 || ^18.0.0
+ dependencies:
+ '@emotion/react': 11.11.0(@types/react@18.2.6)(react@18.2.0)
+ '@emotion/styled': 11.11.0(@emotion/react@11.11.0)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/icons-material': 5.11.16(@mui/material@5.13.1)(@types/react@18.2.6)(react@18.2.0)
+ '@mui/material': 5.13.1(@emotion/react@11.11.0)(@emotion/styled@11.11.0)(@types/react@18.2.6)(react-dom@18.2.0)(react@18.2.0)
+ history: 5.3.0
+ ra-core: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ ra-i18n-polyglot: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ ra-language-english: 4.10.3(history@5.3.0)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ ra-ui-materialui: 4.10.3(@mui/icons-material@5.11.16)(@mui/material@5.13.1)(ra-core@4.10.3)(react-dom@18.2.0)(react-hook-form@7.43.9)(react-router-dom@6.11.2)(react-router@6.11.2)(react@18.2.0)
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ react-hook-form: 7.43.9(react@18.2.0)
+ react-router: 6.11.2(react@18.2.0)
+ react-router-dom: 6.11.2(react-dom@18.2.0)(react@18.2.0)
+ transitivePeerDependencies:
+ - '@types/react'
+ - react-native
+ dev: false
+
+ /react-dom@18.2.0(react@18.2.0):
+ resolution: {integrity: sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==}
+ peerDependencies:
+ react: ^18.2.0
+ dependencies:
+ loose-envify: 1.4.0
+ react: 18.2.0
+ scheduler: 0.23.0
+
+ /react-dropzone@12.1.0(react@18.2.0):
+ resolution: {integrity: sha512-iBYHA1rbopIvtzokEX4QubO6qk5IF/x3BtKGu74rF2JkQDXnwC4uO/lHKpaw4PJIV6iIAYOlwLv2FpiGyqHNog==}
+ engines: {node: '>= 10.13'}
+ peerDependencies:
+ react: '>= 16.8'
+ dependencies:
+ attr-accept: 2.2.2
+ file-selector: 0.5.0
+ prop-types: 15.8.1
+ react: 18.2.0
+ dev: false
+
+ /react-error-boundary@3.1.4(react@18.2.0):
+ resolution: {integrity: sha512-uM9uPzZJTF6wRQORmSrvOIgt4lJ9MC1sNgEOj2XGsDTRE4kmpWxg7ENK9EWNKJRMAOY9z0MuF4yIfl6gp4sotA==}
+ engines: {node: '>=10', npm: '>=6'}
+ peerDependencies:
+ react: '>=16.13.1'
+ dependencies:
+ '@babel/runtime': 7.21.5
+ react: 18.2.0
+ dev: false
+
+ /react-error-overlay@6.0.11:
+ resolution: {integrity: sha512-/6UZ2qgEyH2aqzYZgQPxEnz33NJ2gNsnHA2o5+o4wW9bLM/JYQitNP9xPhsXwC08hMMovfGe/8retsdDsczPRg==}
+ dev: true
+
+ /react-hook-form@7.43.9(react@18.2.0):
+ resolution: {integrity: sha512-AUDN3Pz2NSeoxQ7Hs6OhQhDr6gtF9YRuutGDwPQqhSUAHJSgGl2VeY3qN19MG0SucpjgDiuMJ4iC5T5uB+eaNQ==}
+ engines: {node: '>=12.22.0'}
+ peerDependencies:
+ react: ^16.8.0 || ^17 || ^18
+ dependencies:
+ react: 18.2.0
+ dev: false
+
+ /react-is@16.13.1:
+ resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==}
+
+ /react-is@17.0.2:
+ resolution: {integrity: sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==}
+
+ /react-is@18.2.0:
+ resolution: {integrity: sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==}
+
+ /react-query@3.39.3(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-nLfLz7GiohKTJDuT4us4X3h/8unOh+00MLb2yJoGTPjxKs2bc1iDhkNx2bd5MKklXnOD3NrVZ+J2UXujA5In4g==}
+ peerDependencies:
+ react: ^16.8.0 || ^17.0.0 || ^18.0.0
+ react-dom: '*'
+ react-native: '*'
+ peerDependenciesMeta:
+ react-dom:
+ optional: true
+ react-native:
+ optional: true
+ dependencies:
+ '@babel/runtime': 7.21.5
+ broadcast-channel: 3.7.0
+ match-sorter: 6.3.1
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ dev: false
+
+ /react-refresh@0.14.0:
+ resolution: {integrity: sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ==}
+ engines: {node: '>=0.10.0'}
+ dev: true
+
+ /react-router-dom@6.11.2(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-JNbKtAeh1VSJQnH6RvBDNhxNwemRj7KxCzc5jb7zvDSKRnPWIFj9pO+eXqjM69gQJ0r46hSz1x4l9y0651DKWw==}
+ engines: {node: '>=14'}
+ peerDependencies:
+ react: '>=16.8'
+ react-dom: '>=16.8'
+ dependencies:
+ '@remix-run/router': 1.6.2
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ react-router: 6.11.2(react@18.2.0)
+ dev: false
+
+ /react-router@6.11.2(react@18.2.0):
+ resolution: {integrity: sha512-74z9xUSaSX07t3LM+pS6Un0T55ibUE/79CzfZpy5wsPDZaea1F8QkrsiyRnA2YQ7LwE/umaydzXZV80iDCPkMg==}
+ engines: {node: '>=14'}
+ peerDependencies:
+ react: '>=16.8'
+ dependencies:
+ '@remix-run/router': 1.6.2
+ react: 18.2.0
+ dev: false
+
+ /react-transition-group@4.4.5(react-dom@18.2.0)(react@18.2.0):
+ resolution: {integrity: sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g==}
+ peerDependencies:
+ react: '>=16.6.0'
+ react-dom: '>=16.6.0'
+ dependencies:
+ '@babel/runtime': 7.21.5
+ dom-helpers: 5.2.1
+ loose-envify: 1.4.0
+ prop-types: 15.8.1
+ react: 18.2.0
+ react-dom: 18.2.0(react@18.2.0)
+ dev: false
+
+ /react@18.2.0:
+ resolution: {integrity: sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==}
+ engines: {node: '>=0.10.0'}
+ dependencies:
+ loose-envify: 1.4.0
+
+ /redent@3.0.0:
+ resolution: {integrity: sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==}
+ engines: {node: '>=8'}
+ dependencies:
+ indent-string: 4.0.0
+ strip-indent: 3.0.0
+ dev: true
+
+ /regenerator-runtime@0.13.11:
+ resolution: {integrity: sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==}
+
+ /regexp.prototype.flags@1.5.0:
+ resolution: {integrity: sha512-0SutC3pNudRKgquxGoRGIz946MZVHqbNfPjBdxeOhBrdgDKlRoXmYLQN9xRbrR09ZXWeGAdPuif7egofn6v5LA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ functions-have-names: 1.2.3
+
+ /remove-accents@0.4.2:
+ resolution: {integrity: sha512-7pXIJqJOq5tFgG1A2Zxti3Ht8jJF337m4sowbuHsW30ZnkQFnDzy9qBNhgzX8ZLW4+UBcXiiR7SwR6pokHsxiA==}
+ dev: false
+
+ /remove-accents@0.4.4:
+ resolution: {integrity: sha512-EpFcOa/ISetVHEXqu+VwI96KZBmq+a8LJnGkaeFw45epGlxIZz5dhEEnNZMsQXgORu3qaMoLX4qJCzOik6ytAg==}
+ dev: false
+
+ /require-directory@2.1.1:
+ resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==}
+ engines: {node: '>=0.10.0'}
+ dev: true
+
+ /resolve-from@4.0.0:
+ resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==}
+ engines: {node: '>=4'}
+
+ /resolve@1.22.2:
+ resolution: {integrity: sha512-Sb+mjNHOULsBv818T40qSPeRiuWLyaGMa5ewydRLFimneixmVy2zdivRl+AF6jaYPC8ERxGDmFSiqui6SfPd+g==}
+ hasBin: true
+ dependencies:
+ is-core-module: 2.12.1
+ path-parse: 1.0.7
+ supports-preserve-symlinks-flag: 1.0.0
+
+ /resolve@2.0.0-next.4:
+ resolution: {integrity: sha512-iMDbmAWtfU+MHpxt/I5iWI7cY6YVEZUQ3MBgPQ++XD1PELuJHIl82xBmObyP2KyQmkNB2dsqF7seoQQiAn5yDQ==}
+ hasBin: true
+ dependencies:
+ is-core-module: 2.12.1
+ path-parse: 1.0.7
+ supports-preserve-symlinks-flag: 1.0.0
+ dev: true
+
+ /reusify@1.0.4:
+ resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==}
+ engines: {iojs: '>=1.0.0', node: '>=0.10.0'}
+ dev: true
+
+ /rimraf@2.6.3:
+ resolution: {integrity: sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==}
+ hasBin: true
+ dependencies:
+ glob: 7.2.3
+ dev: true
+
+ /rimraf@3.0.2:
+ resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==}
+ hasBin: true
+ dependencies:
+ glob: 7.2.3
+
+ /rollup@3.23.0:
+ resolution: {integrity: sha512-h31UlwEi7FHihLe1zbk+3Q7z1k/84rb9BSwmBSr/XjOCEaBJ2YyedQDuM0t/kfOS0IxM+vk1/zI9XxYj9V+NJQ==}
+ engines: {node: '>=14.18.0', npm: '>=8.0.0'}
+ hasBin: true
+ optionalDependencies:
+ fsevents: 2.3.2
+ dev: true
+
+ /rope-sequence@1.3.4:
+ resolution: {integrity: sha512-UT5EDe2cu2E/6O4igUr5PSFs23nvvukicWHx6GnOPlHAiiYbzNuCRQCuiUdHJQcqKalLKlrYJnjY0ySGsXNQXQ==}
+ dev: false
+
+ /run-parallel@1.2.0:
+ resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==}
+ dependencies:
+ queue-microtask: 1.2.3
+ dev: true
+
+ /safe-regex-test@1.0.0:
+ resolution: {integrity: sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==}
+ dependencies:
+ call-bind: 1.0.2
+ get-intrinsic: 1.2.1
+ is-regex: 1.1.4
+
+ /scheduler@0.23.0:
+ resolution: {integrity: sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==}
+ dependencies:
+ loose-envify: 1.4.0
+
+ /semver@6.3.0:
+ resolution: {integrity: sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==}
+ hasBin: true
+
+ /shebang-command@2.0.0:
+ resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==}
+ engines: {node: '>=8'}
+ dependencies:
+ shebang-regex: 3.0.0
+ dev: true
+
+ /shebang-regex@3.0.0:
+ resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /side-channel@1.0.4:
+ resolution: {integrity: sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==}
+ dependencies:
+ call-bind: 1.0.2
+ get-intrinsic: 1.2.1
+ object-inspect: 1.12.3
+
+ /slash@3.0.0:
+ resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /source-map-explorer@2.5.3:
+ resolution: {integrity: sha512-qfUGs7UHsOBE5p/lGfQdaAj/5U/GWYBw2imEpD6UQNkqElYonkow8t+HBL1qqIl3CuGZx7n8/CQo4x1HwSHhsg==}
+ engines: {node: '>=12'}
+ hasBin: true
+ dependencies:
+ btoa: 1.2.1
+ chalk: 4.1.2
+ convert-source-map: 1.9.0
+ ejs: 3.1.9
+ escape-html: 1.0.3
+ glob: 7.2.3
+ gzip-size: 6.0.0
+ lodash: 4.17.21
+ open: 7.4.2
+ source-map: 0.7.4
+ temp: 0.9.4
+ yargs: 16.2.0
+ dev: true
+
+ /source-map-js@1.0.2:
+ resolution: {integrity: sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==}
+ engines: {node: '>=0.10.0'}
+ dev: true
+
+ /source-map@0.5.7:
+ resolution: {integrity: sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==}
+ engines: {node: '>=0.10.0'}
+ dev: false
+
+ /source-map@0.7.4:
+ resolution: {integrity: sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==}
+ engines: {node: '>= 8'}
+ dev: true
+
+ /split-on-first@1.1.0:
+ resolution: {integrity: sha512-43ZssAJaMusuKWL8sKUBQXHWOpq8d6CfN/u1p4gUzfJkM05C8rxTmYrkIPTXapZpORA6LkkzcUulJ8FqA7Uudw==}
+ engines: {node: '>=6'}
+ dev: false
+
+ /split-on-first@3.0.0:
+ resolution: {integrity: sha512-qxQJTx2ryR0Dw0ITYyekNQWpz6f8dGd7vffGNflQQ3Iqj9NJ6qiZ7ELpZsJ/QBhIVAiDfXdag3+Gp8RvWa62AA==}
+ engines: {node: '>=12'}
+ dev: false
+
+ /stack-utils@2.0.6:
+ resolution: {integrity: sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==}
+ engines: {node: '>=10'}
+ dependencies:
+ escape-string-regexp: 2.0.0
+ dev: true
+
+ /stop-iteration-iterator@1.0.0:
+ resolution: {integrity: sha512-iCGQj+0l0HOdZ2AEeBADlsRC+vsnDsZsbdSiH1yNSjcfKM7fdpCMfqAL/dwF5BLiw/XhRft/Wax6zQbhq2BcjQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ internal-slot: 1.0.5
+ dev: true
+
+ /strict-uri-encode@2.0.0:
+ resolution: {integrity: sha512-QwiXZgpRcKkhTj2Scnn++4PKtWsH0kpzZ62L2R6c/LUVYv7hVnZqcg2+sMuT6R7Jusu1vviK/MFsu6kNJfWlEQ==}
+ engines: {node: '>=4'}
+ dev: false
+
+ /string-width@4.2.3:
+ resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==}
+ engines: {node: '>=8'}
+ dependencies:
+ emoji-regex: 8.0.0
+ is-fullwidth-code-point: 3.0.0
+ strip-ansi: 6.0.1
+ dev: true
+
+ /string.prototype.matchall@4.0.8:
+ resolution: {integrity: sha512-6zOCOcJ+RJAQshcTvXPHoxoQGONa3e/Lqx90wUA+wEzX78sg5Bo+1tQo4N0pohS0erG9qtCqJDjNCQBjeWVxyg==}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+ get-intrinsic: 1.2.1
+ has-symbols: 1.0.3
+ internal-slot: 1.0.5
+ regexp.prototype.flags: 1.5.0
+ side-channel: 1.0.4
+ dev: true
+
+ /string.prototype.trim@1.2.7:
+ resolution: {integrity: sha512-p6TmeT1T3411M8Cgg9wBTMRtY2q9+PNy9EV1i2lIXUN/btt763oIfxwN3RR8VU6wHX8j/1CFy0L+YuThm6bgOg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+
+ /string.prototype.trimend@1.0.6:
+ resolution: {integrity: sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+
+ /string.prototype.trimstart@1.0.6:
+ resolution: {integrity: sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==}
+ dependencies:
+ call-bind: 1.0.2
+ define-properties: 1.2.0
+ es-abstract: 1.21.2
+
+ /strip-ansi@6.0.1:
+ resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==}
+ engines: {node: '>=8'}
+ dependencies:
+ ansi-regex: 5.0.1
+ dev: true
+
+ /strip-bom@3.0.0:
+ resolution: {integrity: sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==}
+ engines: {node: '>=4'}
+ dev: true
+
+ /strip-indent@3.0.0:
+ resolution: {integrity: sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==}
+ engines: {node: '>=8'}
+ dependencies:
+ min-indent: 1.0.1
+ dev: true
+
+ /strip-json-comments@3.1.1:
+ resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==}
+ engines: {node: '>=8'}
+ dev: true
+
+ /stylis@4.2.0:
+ resolution: {integrity: sha512-Orov6g6BB1sDfYgzWfTHDOxamtX1bE/zo104Dh9e6fqJ3PooipYyfJ0pUmrZO2wAvO8YbEyeFrkV91XTsGMSrw==}
+ dev: false
+
+ /supports-color@5.5.0:
+ resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==}
+ engines: {node: '>=4'}
+ dependencies:
+ has-flag: 3.0.0
+
+ /supports-color@7.2.0:
+ resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==}
+ engines: {node: '>=8'}
+ dependencies:
+ has-flag: 4.0.0
+ dev: true
+
+ /supports-preserve-symlinks-flag@1.0.0:
+ resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==}
+ engines: {node: '>= 0.4'}
+
+ /temp@0.9.4:
+ resolution: {integrity: sha512-yYrrsWnrXMcdsnu/7YMYAofM1ktpL5By7vZhf15CrXijWWrEYZks5AXBudalfSWJLlnen/QUJUB5aoB0kqZUGA==}
+ engines: {node: '>=6.0.0'}
+ dependencies:
+ mkdirp: 0.5.6
+ rimraf: 2.6.3
+ dev: true
+
+ /text-table@0.2.0:
+ resolution: {integrity: sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==}
+ dev: true
+
+ /throttle-debounce@3.0.1:
+ resolution: {integrity: sha512-dTEWWNu6JmeVXY0ZYoPuH5cRIwc0MeGbJwah9KUNYSJwommQpCzTySTpEe8Gs1J23aeWEuAobe4Ag7EHVt/LOg==}
+ engines: {node: '>=10'}
+ dev: false
+
+ /tippy.js@6.3.7:
+ resolution: {integrity: sha512-E1d3oP2emgJ9dRQZdf3Kkn0qJgI6ZLpyS5z6ZkY1DF3kaQaBsGZsndEpHwx+eC+tYM41HaSNvNtLx8tU57FzTQ==}
+ dependencies:
+ '@popperjs/core': 2.11.7
+ dev: false
+
+ /to-fast-properties@2.0.0:
+ resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==}
+ engines: {node: '>=4'}
+
+ /to-regex-range@5.0.1:
+ resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==}
+ engines: {node: '>=8.0'}
+ dependencies:
+ is-number: 7.0.0
+ dev: true
+
+ /tsconfig-paths@3.14.2:
+ resolution: {integrity: sha512-o/9iXgCYc5L/JxCHPe3Hvh8Q/2xm5Z+p18PESBU6Ff33695QnCHBEjcytY2q19ua7Mbl/DavtBOLq+oG0RCL+g==}
+ dependencies:
+ '@types/json5': 0.0.29
+ json5: 1.0.2
+ minimist: 1.2.8
+ strip-bom: 3.0.0
+ dev: true
+
+ /tslib@2.5.2:
+ resolution: {integrity: sha512-5svOrSA2w3iGFDs1HibEVBGbDrAY82bFQ3HZ3ixB+88nsbsWQoKqDRb5UBYAUPEzbBn6dAp5gRNXglySbx1MlA==}
+ dev: false
+
+ /type-check@0.4.0:
+ resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==}
+ engines: {node: '>= 0.8.0'}
+ dependencies:
+ prelude-ls: 1.2.1
+ dev: true
+
+ /type-fest@0.20.2:
+ resolution: {integrity: sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==}
+ engines: {node: '>=10'}
+ dev: true
+
+ /type-fest@2.19.0:
+ resolution: {integrity: sha512-RAH822pAdBgcNMAfWnCBU3CFZcfZ/i1eZjwFU/dsLKumyuuP3niueg2UAukXYF0E2AAoc82ZSSf9J0WQBinzHA==}
+ engines: {node: '>=12.20'}
+ dev: false
+
+ /typed-array-length@1.0.4:
+ resolution: {integrity: sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==}
+ dependencies:
+ call-bind: 1.0.2
+ for-each: 0.3.3
+ is-typed-array: 1.1.10
+
+ /typescript@5.0.4:
+ resolution: {integrity: sha512-cW9T5W9xY37cc+jfEnaUvX91foxtHkza3Nw3wkoF4sSlKn0MONdkdEndig/qPBWXNkmplh3NzayQzCiHM4/hqw==}
+ engines: {node: '>=12.20'}
+ hasBin: true
+ dev: true
+
+ /uc.micro@1.0.6:
+ resolution: {integrity: sha512-8Y75pvTYkLJW2hWQHXxoqRgV7qb9B+9vFEtidML+7koHUFapnVJAZ6cKs+Qjz5Aw3aZWHMC6u0wJE3At+nSGwA==}
+ dev: false
+
+ /unbox-primitive@1.0.2:
+ resolution: {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
+
+ /unload@2.2.0:
+ resolution: {integrity: sha512-B60uB5TNBLtN6/LsgAf3udH9saB5p7gqJwcFfbOEZ8BcBHnGwCf6G/TGiEqkRAxX7zAFIUtzdrXQSdL3Q/wqNA==}
+ dependencies:
+ '@babel/runtime': 7.21.5
+ detect-node: 2.1.0
+ dev: false
+
+ /update-browserslist-db@1.0.11(browserslist@4.21.5):
+ resolution: {integrity: sha512-dCwEFf0/oT85M1fHBg4F0jtLwJrutGoHSQXCh7u4o2t1drG+c0a9Flnqww6XUKSfQMPpJBRjU8d4RXB09qtvaA==}
+ hasBin: true
+ peerDependencies:
+ browserslist: '>= 4.21.0'
+ dependencies:
+ browserslist: 4.21.5
+ escalade: 3.1.1
+ picocolors: 1.0.0
+
+ /uri-js@4.4.1:
+ resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==}
+ dependencies:
+ punycode: 2.3.0
+ dev: true
+
+ /vite@4.3.8:
+ resolution: {integrity: sha512-uYB8PwN7hbMrf4j1xzGDk/lqjsZvCDbt/JC5dyfxc19Pg8kRm14LinK/uq+HSLNswZEoKmweGdtpbnxRtrAXiQ==}
+ engines: {node: ^14.18.0 || >=16.0.0}
+ hasBin: true
+ peerDependencies:
+ '@types/node': '>= 14'
+ less: '*'
+ sass: '*'
+ stylus: '*'
+ sugarss: '*'
+ terser: ^5.4.0
+ peerDependenciesMeta:
+ '@types/node':
+ optional: true
+ less:
+ optional: true
+ sass:
+ optional: true
+ stylus:
+ optional: true
+ sugarss:
+ optional: true
+ terser:
+ optional: true
+ dependencies:
+ esbuild: 0.17.19
+ postcss: 8.4.23
+ rollup: 3.23.0
+ optionalDependencies:
+ fsevents: 2.3.2
+ dev: true
+
+ /w3c-keyname@2.2.7:
+ resolution: {integrity: sha512-XB8aa62d4rrVfoZYQaYNy3fy+z4nrfy2ooea3/0BnBzXW0tSdZ+lRgjzBZhk0La0H6h8fVyYCxx/qkQcAIuvfg==}
+ dev: false
+
+ /warning@4.0.3:
+ resolution: {integrity: sha512-rpJyN222KWIvHJ/F53XSZv0Zl/accqHR8et1kpaMTD/fLCRxtV8iX8czMzY7sVZupTI3zcUTg8eycS2kNF9l6w==}
+ dependencies:
+ loose-envify: 1.4.0
+ dev: false
+
+ /web-vitals@3.3.1:
+ resolution: {integrity: sha512-LTfY5GjcY3ngFzNsYFSYL+AmVmlWrzPTUxSMDis2rZbf+SzT7HH3NH4Y/l45XOlrAIunOBeURN9qtBHkRskAiA==}
+ dev: false
+
+ /which-boxed-primitive@1.0.2:
+ resolution: {integrity: sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==}
+ dependencies:
+ is-bigint: 1.0.4
+ is-boolean-object: 1.1.2
+ is-number-object: 1.0.7
+ is-string: 1.0.7
+ is-symbol: 1.0.4
+
+ /which-collection@1.0.1:
+ resolution: {integrity: sha512-W8xeTUwaln8i3K/cY1nGXzdnVZlidBcagyNFtBdD5kxnb4TvGKR7FfSIS3mYpwWS1QUCutfKz8IY8RjftB0+1A==}
+ dependencies:
+ is-map: 2.0.2
+ is-set: 2.0.2
+ is-weakmap: 2.0.1
+ is-weakset: 2.0.2
+ dev: true
+
+ /which-typed-array@1.1.9:
+ resolution: {integrity: sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA==}
+ engines: {node: '>= 0.4'}
+ 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
+ is-typed-array: 1.1.10
+
+ /which@2.0.2:
+ resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==}
+ engines: {node: '>= 8'}
+ hasBin: true
+ dependencies:
+ isexe: 2.0.0
+ dev: true
+
+ /word-wrap@1.2.3:
+ resolution: {integrity: sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==}
+ engines: {node: '>=0.10.0'}
+ dev: true
+
+ /wrap-ansi@7.0.0:
+ resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==}
+ engines: {node: '>=10'}
+ dependencies:
+ ansi-styles: 4.3.0
+ string-width: 4.2.3
+ strip-ansi: 6.0.1
+ dev: true
+
+ /wrappy@1.0.2:
+ resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==}
+
+ /y18n@5.0.8:
+ resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==}
+ engines: {node: '>=10'}
+ dev: true
+
+ /yallist@3.1.1:
+ resolution: {integrity: sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==}
+
+ /yaml@1.10.2:
+ resolution: {integrity: sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==}
+ engines: {node: '>= 6'}
+ dev: false
+
+ /yargs-parser@20.2.9:
+ resolution: {integrity: sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==}
+ engines: {node: '>=10'}
+ dev: true
+
+ /yargs@16.2.0:
+ resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==}
+ engines: {node: '>=10'}
+ dependencies:
+ cliui: 7.0.4
+ escalade: 3.1.1
+ get-caller-file: 2.0.5
+ require-directory: 2.1.1
+ string-width: 4.2.3
+ y18n: 5.0.8
+ yargs-parser: 20.2.9
+ dev: true
+
+ /yocto-queue@0.1.0:
+ resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==}
+ engines: {node: '>=10'}
+ dev: true
diff --git a/frontend/admin/public/logo.svg b/frontend/admin/public/logo.svg
new file mode 100644
index 0000000..4723244
--- /dev/null
+++ b/frontend/admin/public/logo.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/admin/public/manifest.json b/frontend/admin/public/manifest.json
new file mode 100644
index 0000000..f73c3bf
--- /dev/null
+++ b/frontend/admin/public/manifest.json
@@ -0,0 +1,15 @@
+{
+ "short_name": "Sergey Rashin",
+ "name": "Sergey Rashin - Portfolio",
+ "icons": [
+ {
+ "src": "logo.svg",
+ "type": "image/svg",
+ "sizes": "835x519"
+ }
+ ],
+ "start_url": ".",
+ "display": "standalone",
+ "theme_color": "#000000",
+ "background_color": "#ffffff"
+}
diff --git a/frontend/admin/public/robots.txt b/frontend/admin/public/robots.txt
new file mode 100644
index 0000000..e9e57dc
--- /dev/null
+++ b/frontend/admin/public/robots.txt
@@ -0,0 +1,3 @@
+# https://www.robotstxt.org/robotstxt.html
+User-agent: *
+Disallow:
diff --git a/frontend/admin/src/App.tsx b/frontend/admin/src/App.tsx
new file mode 100644
index 0000000..a674258
--- /dev/null
+++ b/frontend/admin/src/App.tsx
@@ -0,0 +1,47 @@
+import {Admin, EditGuesser, Layout, ListGuesser, Resource, CustomRoutes} from "react-admin";
+import dataProvider from "./dataProvider";
+import {authProvider} from "./authProvider";
+import { Route } from "react-router-dom";
+
+
+import projects from "./pages/projects";
+import jobs from "./pages/jobs";
+import sections from "./pages/sections";
+import skills from "./pages/skills";
+import educations from "./pages/educations";
+
+import {
+ Group as UserIcon,
+} from "@mui/icons-material";
+import MyMenu from "./components/MyMenu";
+import Configuration from "./pages/configuration/Configuration";
+
+const MyLayout = (props: any) =>
+
+const App = () => {
+ return (
+
+ {/**/}
+ {/**/}
+
+
+
+
+
+
+
+ } />
+
+ {/**/}
+
+
+ );
+};
+export default App;
diff --git a/frontend/admin/src/addUploadFeature.ts b/frontend/admin/src/addUploadFeature.ts
new file mode 100644
index 0000000..a31714e
--- /dev/null
+++ b/frontend/admin/src/addUploadFeature.ts
@@ -0,0 +1,54 @@
+import {DataProvider} from "ra-core/src/types";
+import {UpdateParams} from "react-admin";
+
+/**
+ * Convert a `File` object returned by the upload input into a base 64 string.
+ * That's not the most optimized way to store images in production, but it's
+ * enough to illustrate the idea of data provider decoration.
+ */
+// const convertFileToBase64 = file => new Promise((resolve, reject) => {
+// const reader = new FileReader();
+// reader.readAsDataURL(file.rawFile);
+//
+// reader.onload = () => resolve(reader.result);
+// reader.onerror = reject;
+// });
+
+const uploadImage = async (dataProvider: DataProvider, resource: string, params: UpdateParams) => {
+ if (resource === 'projects' || resource === 'skills') {
+ if (params.data.hasOwnProperty('image')) {
+ const imageData = params.data.image;
+ delete params.data.image;
+
+ if (imageData != null && !imageData.hasOwnProperty('id')) {
+ let formData = new FormData();
+ let rawFile = imageData.rawFile;
+ formData.append('file', rawFile, rawFile.name);
+
+ const result = await dataProvider.sendForm("storage", formData);
+
+ params.data.imageId = result.data.id;
+ }
+ else {
+ params.data.imageId = imageData?.id;
+ }
+ }
+ }
+}
+
+const addUploadFeature = (dataProvider: DataProvider) => ({
+ ...dataProvider,
+ create: async (resource: string, params: UpdateParams) => {
+ await uploadImage(dataProvider, resource, params);
+
+ return await dataProvider.create(resource, params);
+ },
+ update: async (resource: string, params: UpdateParams) => {
+ await uploadImage(dataProvider, resource, params);
+
+ return await dataProvider.update(resource, params);
+ },
+});
+
+
+export default addUploadFeature;
diff --git a/frontend/admin/src/authProvider.ts b/frontend/admin/src/authProvider.ts
new file mode 100644
index 0000000..6fbd263
--- /dev/null
+++ b/frontend/admin/src/authProvider.ts
@@ -0,0 +1,66 @@
+const apiUrl = process.env.API_URL ?? 'https://rashin.me';
+
+// @ts-ignore
+export const authProvider = {
+ login: ({ username, password }: {username: string, password: string}) => {
+ const request = new Request(`${apiUrl}/login`, {
+ method: 'POST',
+ body: JSON.stringify({ email: username, password }),
+ headers: new Headers({ 'Content-Type': 'application/json' }),
+ });
+
+ return fetch(request)
+ .then(response => response.json())
+ .then(response => {
+ const isSuccess = response.code === 200;
+
+ if (!isSuccess) {
+ return Promise.reject(response.error);
+ }
+
+ const request = new Request(`${apiUrl}/user`, {
+ method: 'GET',
+ headers: new Headers({ 'Content-Type': 'application/json' }),
+ });
+
+ return fetch(request)
+ .then(response => response.json())
+ .then(response => {
+ localStorage.setItem('auth', JSON.stringify(response));
+ })
+ .catch(res => {
+ return Promise.reject(res);
+ });
+ });
+ },
+ logout: () => {
+ localStorage.removeItem('auth');
+ return Promise.resolve();
+ },
+ checkAuth: () =>
+ localStorage.getItem('auth') ? Promise.resolve() : Promise.reject(),
+ checkError: (error: any) => {
+ const status = error.status;
+ if (status === 401 || status === 403) {
+ localStorage.removeItem('auth');
+ return Promise.reject();
+ }
+
+ return Promise.resolve();
+ },
+ getIdentity: () => {
+ try {
+ const value = localStorage?.getItem('auth')??'';
+ const { id, firstName, lastName } = JSON.parse(value);
+ const fullName = `${firstName} ${lastName}`;
+ return Promise.resolve({ id, fullName });
+ } catch (error) {
+ return Promise.reject(error);
+ }
+ // Promise.resolve({
+ // id: 'user',
+ // fullName: 'John Doe',
+ // })
+ },
+ getPermissions: () => Promise.resolve(''),
+};
diff --git a/frontend/admin/src/components/MyMenu.tsx b/frontend/admin/src/components/MyMenu.tsx
new file mode 100644
index 0000000..b52db5d
--- /dev/null
+++ b/frontend/admin/src/components/MyMenu.tsx
@@ -0,0 +1,19 @@
+import { Menu } from 'react-admin';
+import SettingsApplicationsIcon from '@mui/icons-material/SettingsApplications';
+import React from "react";
+
+const MyMenu = () => {
+ return (
+
+ );
+}
+
+export default MyMenu;
diff --git a/frontend/admin/src/css/education.css b/frontend/admin/src/css/education.css
new file mode 100644
index 0000000..50cb971
--- /dev/null
+++ b/frontend/admin/src/css/education.css
@@ -0,0 +1,19 @@
+.vertical {
+ justify-content: center;
+}
+
+.vertical_alternating {
+ justify-content: center;
+}
+
+@media only screen and (max-width: 578px) {
+ div.vertical-item-row > div:nth-child(1) {
+ width: 20% !important;
+ }
+ div.vertical-item-row > div:nth-child(2) {
+ width: 65% !important;
+ }
+ div.vertical-item-row > div:nth-child(3) {
+ width: 15% !important;
+ }
+}
diff --git a/frontend/admin/src/css/experience.css b/frontend/admin/src/css/experience.css
new file mode 100644
index 0000000..e31c2c9
--- /dev/null
+++ b/frontend/admin/src/css/experience.css
@@ -0,0 +1,9 @@
+.entry .body{
+ margin: 0 0 1em;
+}
+
+@media only screen and (max-width: 768px) {
+ .item-title {
+ margin-top: 10px !important;
+ }
+}
diff --git a/frontend/admin/src/dataProvider.ts b/frontend/admin/src/dataProvider.ts
new file mode 100644
index 0000000..4fa66af
--- /dev/null
+++ b/frontend/admin/src/dataProvider.ts
@@ -0,0 +1,129 @@
+import {fetchUtils, UpdateParams} from "react-admin";
+import queryString from "query-string";
+import {DataProvider} from "ra-core/src/types";
+import {Options} from "ra-core/src/dataProvider/fetch";
+import addUploadFeature from "./addUploadFeature";
+
+const apiUrl = process.env.API_URL ?? 'https://rashin.me';
+
+const httpClient = (url: string, options: Options = {}) => {
+ if (!options.headers) {
+ options.headers = new Headers({
+ Accept: 'application/json'
+ });
+ }
+ // options.credentials = 'include';
+ return fetchUtils.fetchJson(url, options);
+}
+
+const dataProvider: DataProvider = {
+ getList: (resource, params) => {
+ const { page, perPage } = params.pagination;
+ const { field, order } = params.sort;
+
+ const query = {
+ sort: JSON.stringify({field: field, order: order}),
+ offset: (page - 1) * perPage,
+ limit: perPage,
+ filter: JSON.stringify(params.filter),
+ };
+
+ const url = `${apiUrl}/${resource}?${queryString.stringify(query)}`;
+
+ return httpClient(url).then(({ json }) => ({
+ data: json.data,
+ total: json.pagination.total,
+ }));
+ },
+
+ getOne: (resource, params) =>
+ httpClient(`${apiUrl}/${resource}/${params.id}`).then(({ json }) => ({
+ data: json,
+ })),
+
+ getMany: (resource, params) => {
+ const query = {
+ filter: JSON.stringify({ id: params.ids }),
+ };
+ const url = `${apiUrl}/${resource}?${queryString.stringify(query)}`;
+ return httpClient(url).then(({ json }) => ({ data: json.data }));
+ },
+
+ getManyReference: (resource, params) => {
+ const { page, perPage } = params.pagination;
+ const { field, order } = params.sort;
+ const query = {
+ sort: JSON.stringify([field, order]),
+ range: JSON.stringify([(page - 1) * perPage, page * perPage - 1]),
+ filter: JSON.stringify({
+ ...params.filter,
+ [params.target]: params.id,
+ }),
+ };
+ const url = `${apiUrl}/${resource}?${queryString.stringify(query)}`;
+
+ return httpClient(url).then(({ json }) => ({
+ data: json.data,
+ total: json.pagination.total,
+ }));
+ },
+
+ update: (resource: string, params: UpdateParams) => {
+ return httpClient(`${apiUrl}/${resource}/${params.id}`, {
+ method: 'PATCH',
+ body: JSON.stringify(params.data),
+ }).then(({ json }) => ({ data: json }))
+ },
+
+ updateMany: (resource, params) => {
+ const query = {
+ filter: JSON.stringify({ id: params.ids}),
+ };
+ return httpClient(`${apiUrl}/${resource}?${queryString.stringify(query)}`, {
+ method: 'PUT',
+ body: JSON.stringify(params.data),
+ }).then(({ json }) => ({ data: json }));
+ },
+
+ create: (resource, params) =>
+ httpClient(`${apiUrl}/${resource}`, {
+ method: 'POST',
+ body: JSON.stringify(params.data),
+ }).then(({ json }) => ({
+ data: { ...params.data, id: json.id },
+ })),
+
+ delete: (resource, params) =>
+ httpClient(`${apiUrl}/${resource}/${params.id}`, {
+ method: 'DELETE',
+ }).then(({ json }) => ({ data: json })),
+
+ deleteMany: (resource, params) => {
+ const query = {
+ filter: JSON.stringify({ id: params.ids}),
+ };
+ return httpClient(`${apiUrl}/${resource}?${queryString.stringify(query)}`, {
+ method: 'DELETE',
+ }).then(({ json }) => ({ data: json }));
+ },
+ sendForm: (resource: string, formData: FormData) =>
+ httpClient(`${apiUrl}/${resource}`, {
+ method: 'POST',
+ body: formData,
+ }).then(({ json }) => ({
+ data: { ...json },
+ })),
+ get: (resource: string) =>
+ httpClient(`${apiUrl}/${resource}`).then(({ json }) => (json)),
+ patch: (resource: string, params) => {
+ console.log(params)
+ return httpClient(`${apiUrl}/${resource}`, {
+ method: 'PATCH',
+ body: JSON.stringify(params),
+ }).then(({ json }) => (json))
+},
+};
+
+const uploadCapableDataProvider = addUploadFeature(dataProvider);
+
+export default uploadCapableDataProvider;
diff --git a/frontend/admin/src/index.css b/frontend/admin/src/index.css
new file mode 100644
index 0000000..ec2585e
--- /dev/null
+++ b/frontend/admin/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/frontend/admin/src/index.tsx b/frontend/admin/src/index.tsx
new file mode 100644
index 0000000..2e12082
--- /dev/null
+++ b/frontend/admin/src/index.tsx
@@ -0,0 +1,15 @@
+import * as ReactDOMClient from 'react-dom/client';
+import App from './App';
+import reportWebVitals from './reportWebVitals';
+
+const container = document.getElementById('root');
+
+// Create a root.
+// @ts-ignore
+const root = ReactDOMClient.createRoot(container);
+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/frontend/admin/src/pages/configuration/Configuration.tsx b/frontend/admin/src/pages/configuration/Configuration.tsx
new file mode 100644
index 0000000..1595145
--- /dev/null
+++ b/frontend/admin/src/pages/configuration/Configuration.tsx
@@ -0,0 +1,153 @@
+import { Card, CardContent } from '@mui/material';
+import {
+ ArrayInput, FileInput, ImageField,
+ required,
+ SimpleForm,
+ SimpleFormIterator,
+ TextInput,
+ Title,
+ useDataProvider
+} from "react-admin";
+import {useEffect, useState} from "react";
+
+interface Property {
+ key: string,
+ value: string,
+}
+
+interface Config {
+ userName: string,
+ roles: string[],
+ about: string,
+ githubLink: string,
+ linkedinLink: string,
+ siteUrl: string,
+ phoneNumber: string,
+ emailAddress: string,
+ userPhoto: object,
+}
+
+const recordDefaults: Config = {
+ userName: '',
+ roles: [],
+ about: '',
+ githubLink: '',
+ linkedinLink: '',
+ siteUrl: '',
+ emailAddress: '',
+ phoneNumber: '',
+ userPhoto: {},
+}
+
+const Configuration = () => {
+ const dataProvider = useDataProvider();
+ const [record, setRecord] = useState(recordDefaults);
+
+ const loadData = () => {
+ dataProvider.get("properties").then((properties: Property[]) => {
+ properties.forEach( (property: Property) => {
+ const key: string = property.key;
+ const value: string = property.value;
+
+ if (record.hasOwnProperty(key)) {
+ if (key === 'roles') {
+ let json = [];
+ try {
+ json = JSON.parse(value);
+ } catch (e) {}
+
+ record[key] = json;
+ }
+ else if (key === "userPhoto") {
+ if (!isNaN(parseInt(value))) {
+ dataProvider.getOne("storage", {id: value}).then(({data}) => {
+ record[key] = data;
+ });
+ }
+ }
+ else {
+ record[key] = value;
+ }
+ }
+ });
+
+ setRecord(record);
+ });
+
+
+ // dataProvider.
+ // get data
+ };
+ // userName: string
+ // roles: json
+ // about: string
+ // githubLink: string
+ // linkedinLink: string
+ // siteUrl
+ // emailAddress
+ // userPhotoId
+
+ useEffect(() => {
+ loadData();
+ }, []);
+
+ const saveConfiguration = async (data: any) => {
+ let dataToSend: Property[] = [];
+
+ // transform data
+ data.roles = JSON.stringify(data.roles);
+
+ if (data.hasOwnProperty('userPhoto')) {
+ // is new File
+ if (data.userPhoto.hasOwnProperty('rawFile'))
+ {
+ let formData = new FormData();
+ let rawFile = data.userPhoto.rawFile;
+ formData.append('file', rawFile, rawFile.name);
+
+ const result = await dataProvider.sendForm("storage", formData);
+
+ data.userPhoto = result.data.id.toString();
+ }
+ else {
+ data.userPhoto = data.userPhoto.id.toString();
+ }
+ }
+
+ for (let key in data) {
+ dataToSend.push({
+ key: key,
+ value: data[key],
+ })
+ }
+
+ dataProvider.patch("properties", {properties: dataToSend});
+ };
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+};
+
+export default Configuration;
diff --git a/frontend/admin/src/pages/educations/EducationCreate.tsx b/frontend/admin/src/pages/educations/EducationCreate.tsx
new file mode 100644
index 0000000..cce9c9c
--- /dev/null
+++ b/frontend/admin/src/pages/educations/EducationCreate.tsx
@@ -0,0 +1,45 @@
+import {Create, required, SelectInput, SimpleForm, TextInput} from 'react-admin';
+import {Grid} from "@mui/material";
+
+export const EducationCreate = () => {
+ const choiceMonths = [
+ {id: 1, name: 'January'},
+ {id: 2, name: 'February'},
+ {id: 3, name: 'March'},
+ {id: 4, name: 'April'},
+ {id: 5, name: 'May'},
+ {id: 6, name: 'June'},
+ {id: 7, name: 'July'},
+ {id: 8, name: 'August'},
+ {id: 9, name: 'September'},
+ {id: 10, name: 'October'},
+ {id: 11, name: 'November'},
+ {id: 12, name: 'December'},
+ ];
+
+ let choiceYears = [];
+
+ for (let i = new Date().getFullYear(); i > 2000; i--) {
+ choiceYears.push({id: i, name: i});
+ }
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/frontend/admin/src/pages/educations/EducationEdit.tsx b/frontend/admin/src/pages/educations/EducationEdit.tsx
new file mode 100644
index 0000000..4cb0201
--- /dev/null
+++ b/frontend/admin/src/pages/educations/EducationEdit.tsx
@@ -0,0 +1,51 @@
+import {
+ Edit,
+ required,
+ SelectInput,
+ SimpleForm,
+ TextInput
+} from 'react-admin';
+import { Grid } from '@mui/material';
+
+export const EducationEdit = () => {
+ const choiceMonths = [
+ {id: 1, name: 'January'},
+ {id: 2, name: 'February'},
+ {id: 3, name: 'March'},
+ {id: 4, name: 'April'},
+ {id: 5, name: 'May'},
+ {id: 6, name: 'June'},
+ {id: 7, name: 'July'},
+ {id: 8, name: 'August'},
+ {id: 9, name: 'September'},
+ {id: 10, name: 'October'},
+ {id: 11, name: 'November'},
+ {id: 12, name: 'December'},
+ ];
+
+ let choiceYears = [];
+
+ for (let i = new Date().getFullYear(); i > 2000; i--) {
+ choiceYears.push({id: i, name: i});
+ }
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/frontend/admin/src/pages/educations/EducationList.tsx b/frontend/admin/src/pages/educations/EducationList.tsx
new file mode 100644
index 0000000..30358f8
--- /dev/null
+++ b/frontend/admin/src/pages/educations/EducationList.tsx
@@ -0,0 +1,12 @@
+import { Datagrid, List, TextField } from 'react-admin';
+
+export const EducationList = () => (
+
+
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/educations/index.tsx b/frontend/admin/src/pages/educations/index.tsx
new file mode 100644
index 0000000..a22a9d9
--- /dev/null
+++ b/frontend/admin/src/pages/educations/index.tsx
@@ -0,0 +1,11 @@
+import Icon from '@mui/icons-material/School';
+import {EducationCreate} from './EducationCreate';
+import {EducationEdit} from './EducationEdit';
+import {EducationList} from './EducationList';
+
+export default {
+ create: EducationCreate,
+ edit: EducationEdit,
+ list: EducationList,
+ icon: Icon,
+};
diff --git a/frontend/admin/src/pages/jobs/JobCreate.tsx b/frontend/admin/src/pages/jobs/JobCreate.tsx
new file mode 100644
index 0000000..623055f
--- /dev/null
+++ b/frontend/admin/src/pages/jobs/JobCreate.tsx
@@ -0,0 +1,56 @@
+import {Create, required, SelectInput, SimpleForm, TextInput} from 'react-admin';
+import {Grid} from "@mui/material";
+
+export const JobCreate = () => {
+ const choiceMonths = [
+ {id: 1, name: 'January'},
+ {id: 2, name: 'February'},
+ {id: 3, name: 'March'},
+ {id: 4, name: 'April'},
+ {id: 5, name: 'May'},
+ {id: 6, name: 'June'},
+ {id: 7, name: 'July'},
+ {id: 8, name: 'August'},
+ {id: 9, name: 'September'},
+ {id: 10, name: 'October'},
+ {id: 11, name: 'November'},
+ {id: 12, name: 'December'},
+ ];
+
+ let choiceYears = [];
+
+ for (let i = new Date().getFullYear(); i > 2000; i--) {
+ choiceYears.push({id: i, name: i});
+ }
+
+ return (
+
+
+
+
+
+
+
+
+
+
+ {/**/}
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/frontend/admin/src/pages/jobs/JobEdit.tsx b/frontend/admin/src/pages/jobs/JobEdit.tsx
new file mode 100644
index 0000000..8522270
--- /dev/null
+++ b/frontend/admin/src/pages/jobs/JobEdit.tsx
@@ -0,0 +1,61 @@
+import {
+ Edit,
+ required,
+ SelectInput,
+ SimpleForm,
+ TextInput
+} from 'react-admin';
+import { Grid } from '@mui/material';
+
+export const JobEdit = () => {
+ const choiceMonths = [
+ {id: 1, name: 'January'},
+ {id: 2, name: 'February'},
+ {id: 3, name: 'March'},
+ {id: 4, name: 'April'},
+ {id: 5, name: 'May'},
+ {id: 6, name: 'June'},
+ {id: 7, name: 'July'},
+ {id: 8, name: 'August'},
+ {id: 9, name: 'September'},
+ {id: 10, name: 'October'},
+ {id: 11, name: 'November'},
+ {id: 12, name: 'December'},
+ ];
+
+ let choiceYears = [];
+
+ for (let i = new Date().getFullYear(); i > 2000; i--) {
+ choiceYears.push({id: i, name: i});
+ }
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/frontend/admin/src/pages/jobs/JobList.tsx b/frontend/admin/src/pages/jobs/JobList.tsx
new file mode 100644
index 0000000..db35450
--- /dev/null
+++ b/frontend/admin/src/pages/jobs/JobList.tsx
@@ -0,0 +1,12 @@
+import { Datagrid, List, TextField } from 'react-admin';
+
+export const JobList = () => (
+
+
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/jobs/index.tsx b/frontend/admin/src/pages/jobs/index.tsx
new file mode 100644
index 0000000..b9e2039
--- /dev/null
+++ b/frontend/admin/src/pages/jobs/index.tsx
@@ -0,0 +1,11 @@
+import Icon from '@mui/icons-material/Work';
+import {JobCreate} from './JobCreate';
+import {JobEdit} from './JobEdit';
+import {JobList} from './JobList';
+
+export default {
+ create: JobCreate,
+ edit: JobEdit,
+ list: JobList,
+ icon: Icon,
+};
diff --git a/frontend/admin/src/pages/projects/ProjectCreate.tsx b/frontend/admin/src/pages/projects/ProjectCreate.tsx
new file mode 100644
index 0000000..bf188a9
--- /dev/null
+++ b/frontend/admin/src/pages/projects/ProjectCreate.tsx
@@ -0,0 +1,47 @@
+import {
+ ArrayInput,
+ Create,
+ FileInput,
+ ImageField,
+ NumberInput,
+ required,
+ SimpleForm,
+ SimpleFormIterator,
+ TextInput,
+ useNotify,
+ useRedirect
+} from 'react-admin';
+
+export const ProjectCreate = () => {
+ const notify = useNotify();
+ const redirect = useRedirect();
+
+ const onSuccess = (data: any) => {
+ notify(`Project created successfully`); // default message is 'ra.notification.created'
+ redirect('edit', 'projects', data.id, data);
+ };
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/frontend/admin/src/pages/projects/ProjectEdit.tsx b/frontend/admin/src/pages/projects/ProjectEdit.tsx
new file mode 100644
index 0000000..f65b815
--- /dev/null
+++ b/frontend/admin/src/pages/projects/ProjectEdit.tsx
@@ -0,0 +1,41 @@
+import {
+ ArrayInput,
+ Edit,
+ FileInput,
+ ImageField,
+ NumberInput,
+ required,
+ SimpleForm,
+ SimpleFormIterator,
+ TextInput, useRecordContext
+} from 'react-admin';
+
+const ProjectTitle = () => {
+ const record = useRecordContext();
+
+ return Edit Project {record ? `"${record.name}"` : ''};
+};
+
+export const ProjectEdit = () => (
+ }>
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/projects/ProjectList.tsx b/frontend/admin/src/pages/projects/ProjectList.tsx
new file mode 100644
index 0000000..4ba83e4
--- /dev/null
+++ b/frontend/admin/src/pages/projects/ProjectList.tsx
@@ -0,0 +1,21 @@
+import {
+ Datagrid,
+ List,
+ TextField,
+ TextInput,
+} from 'react-admin';
+
+
+const postFilters = [
+ ,
+];
+
+// @ts-ignore
+export const ProjectList = () => (
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/projects/index.tsx b/frontend/admin/src/pages/projects/index.tsx
new file mode 100644
index 0000000..d4efc3e
--- /dev/null
+++ b/frontend/admin/src/pages/projects/index.tsx
@@ -0,0 +1,11 @@
+import Icon from '@mui/icons-material/Book';
+import {ProjectList} from './ProjectList';
+import {ProjectEdit} from './ProjectEdit';
+import {ProjectCreate} from './ProjectCreate';
+
+export default {
+ list: ProjectList,
+ create: ProjectCreate,
+ edit: ProjectEdit,
+ icon: Icon,
+};
diff --git a/frontend/admin/src/pages/sections/SectionCreate.tsx b/frontend/admin/src/pages/sections/SectionCreate.tsx
new file mode 100644
index 0000000..9a1b044
--- /dev/null
+++ b/frontend/admin/src/pages/sections/SectionCreate.tsx
@@ -0,0 +1,9 @@
+import { Create, SimpleForm, TextInput } from 'react-admin';
+
+export const SectionCreate = () => (
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/sections/SectionEdit.tsx b/frontend/admin/src/pages/sections/SectionEdit.tsx
new file mode 100644
index 0000000..383e038
--- /dev/null
+++ b/frontend/admin/src/pages/sections/SectionEdit.tsx
@@ -0,0 +1,9 @@
+import { Edit, SimpleForm, TextInput } from 'react-admin';
+
+export const SectionEdit = () => (
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/sections/SectionList.tsx b/frontend/admin/src/pages/sections/SectionList.tsx
new file mode 100644
index 0000000..514bcbf
--- /dev/null
+++ b/frontend/admin/src/pages/sections/SectionList.tsx
@@ -0,0 +1,10 @@
+import { Datagrid, List, TextField } from 'react-admin';
+
+export const SectionList = () => (
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/sections/index.tsx b/frontend/admin/src/pages/sections/index.tsx
new file mode 100644
index 0000000..3fb239b
--- /dev/null
+++ b/frontend/admin/src/pages/sections/index.tsx
@@ -0,0 +1,11 @@
+import Icon from '@mui/icons-material/Category';
+import {SectionList} from './SectionList';
+import {SectionEdit} from './SectionEdit';
+import {SectionCreate} from './SectionCreate';
+
+export default {
+ list: SectionList,
+ create: SectionCreate,
+ edit: SectionEdit,
+ icon: Icon,
+};
diff --git a/frontend/admin/src/pages/skills/SkillCreate.tsx b/frontend/admin/src/pages/skills/SkillCreate.tsx
new file mode 100644
index 0000000..c880ff2
--- /dev/null
+++ b/frontend/admin/src/pages/skills/SkillCreate.tsx
@@ -0,0 +1,25 @@
+import {
+ AutocompleteInput,
+ Create,
+ FileInput,
+ ImageField,
+ ReferenceInput,
+ required,
+ SimpleForm,
+ TextInput
+} from 'react-admin';
+
+export const SkillCreate = () => (
+
+
+
+
+
+
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/skills/SkillEdit.tsx b/frontend/admin/src/pages/skills/SkillEdit.tsx
new file mode 100644
index 0000000..b679948
--- /dev/null
+++ b/frontend/admin/src/pages/skills/SkillEdit.tsx
@@ -0,0 +1,25 @@
+import {
+ AutocompleteInput,
+ Edit,
+ FileInput,
+ ImageField,
+ ReferenceInput,
+ required,
+ SimpleForm,
+ TextInput
+} from 'react-admin';
+
+export const SkillEdit = () => (
+
+
+
+
+
+
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/skills/SkillList.tsx b/frontend/admin/src/pages/skills/SkillList.tsx
new file mode 100644
index 0000000..fae02b6
--- /dev/null
+++ b/frontend/admin/src/pages/skills/SkillList.tsx
@@ -0,0 +1,14 @@
+import {
+ Datagrid,
+ List,
+ TextField,
+} from 'react-admin';
+
+export const SkillList = () => (
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/skills/index.tsx b/frontend/admin/src/pages/skills/index.tsx
new file mode 100644
index 0000000..d616a08
--- /dev/null
+++ b/frontend/admin/src/pages/skills/index.tsx
@@ -0,0 +1,11 @@
+import Icon from '@mui/icons-material/Code';
+import {SkillList} from './SkillList';
+import {SkillEdit} from './SkillEdit';
+import {SkillCreate} from './SkillCreate';
+
+export default {
+ list: SkillList,
+ create: SkillCreate,
+ edit: SkillEdit,
+ icon: Icon,
+};
diff --git a/frontend/admin/src/pages/users/UserList.tsx b/frontend/admin/src/pages/users/UserList.tsx
new file mode 100644
index 0000000..a4aa5f0
--- /dev/null
+++ b/frontend/admin/src/pages/users/UserList.tsx
@@ -0,0 +1,12 @@
+import { Datagrid, EmailField, List, TextField } from 'react-admin';
+
+export const UserList = () => (
+
+
+
+
+
+
+
+
+);
diff --git a/frontend/admin/src/pages/users/index.tsx b/frontend/admin/src/pages/users/index.tsx
new file mode 100644
index 0000000..e69de29
diff --git a/frontend/admin/src/reportWebVitals.ts b/frontend/admin/src/reportWebVitals.ts
new file mode 100644
index 0000000..2db41a8
--- /dev/null
+++ b/frontend/admin/src/reportWebVitals.ts
@@ -0,0 +1,17 @@
+import {ReportHandler} from "web-vitals";
+
+const reportWebVitals = (onPerfEntry?: ReportHandler) => {
+ 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/frontend/admin/src/setupTests.js b/frontend/admin/src/setupTests.js
new file mode 100644
index 0000000..8f2609b
--- /dev/null
+++ b/frontend/admin/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/frontend/admin/src/theme/GlobalStyles.js b/frontend/admin/src/theme/GlobalStyles.js
new file mode 100644
index 0000000..132c225
--- /dev/null
+++ b/frontend/admin/src/theme/GlobalStyles.js
@@ -0,0 +1,11 @@
+import { createGlobalStyle } from 'styled-components';
+
+const GlobalStyles = createGlobalStyle`
+ body {
+ background: ${({ theme }) => theme.background};
+ color: ${({ theme }) => theme.color};
+ transition: all 0.50s linear;
+ }
+`;
+
+export default GlobalStyles;
diff --git a/frontend/admin/src/theme/themes.js b/frontend/admin/src/theme/themes.js
new file mode 100644
index 0000000..783aa51
--- /dev/null
+++ b/frontend/admin/src/theme/themes.js
@@ -0,0 +1,47 @@
+export const lightTheme = {
+ background: '#fff',
+ color: '#121212',
+ accentColor: '#3D84C6',
+ chronoTheme: {
+ cardBgColor: 'white',
+ cardForeColor: 'black',
+ titleColor: 'black',
+ titleColorActive: 'black',
+ },
+ timelineLineColor: '#ccc',
+ cardBackground: '#fff',
+ cardFooterBackground: '#f7f7f7',
+ cardBorderColor: '#00000020',
+ navbarTheme: {
+ linkColor: '#dedede',
+ linkHoverColor: '#fefefe',
+ linkActiveColor: '#fefefe',
+ },
+ bsPrimaryVariant: 'light',
+ bsSecondaryVariant: 'dark',
+ socialIconBgColor: '#121212',
+};
+
+export const darkTheme = {
+ background: '#121212',
+ color: '#eee',
+ accentColor: '#3D84C6',
+ chronoTheme: {
+ cardBgColor: '#1B1B1B',
+ cardForeColor: '#eee',
+ titleColor: 'white',
+ titleColorActive: 'black',
+ },
+ timelineLineColor: '#444',
+ cardBackground: '#060606',
+ cardFooterBackground: '#181818',
+ cardBorderColor: '#ffffff20',
+ navbarTheme: {
+ linkColor: '#dedede',
+ linkHoverColor: '#fefefe',
+ linkActiveColor: '#fefefe',
+ },
+ bsPrimaryVariant: 'dark',
+ bsSecondaryVariant: 'light',
+ socialIconBgColor: '#fefefe',
+};
diff --git a/frontend/admin/tsconfig.json b/frontend/admin/tsconfig.json
new file mode 100644
index 0000000..30d6ff1
--- /dev/null
+++ b/frontend/admin/tsconfig.json
@@ -0,0 +1,25 @@
+{
+ "compilerOptions": {
+ "target": "ES2020",
+ "useDefineForClassFields": true,
+ "lib": ["ES2020", "DOM", "DOM.Iterable"],
+ "module": "ESNext",
+ "skipLibCheck": true,
+
+ /* Bundler mode */
+ "moduleResolution": "bundler",
+ "allowImportingTsExtensions": true,
+ "resolveJsonModule": true,
+ "isolatedModules": true,
+ "noEmit": true,
+ "jsx": "react-jsx",
+
+ /* Linting */
+ "strict": true,
+ "noUnusedLocals": true,
+ "noUnusedParameters": true,
+ "noFallthroughCasesInSwitch": true
+ },
+ "include": ["src"],
+ "references": [{ "path": "./tsconfig.node.json" }]
+}
diff --git a/frontend/admin/tsconfig.node.json b/frontend/admin/tsconfig.node.json
new file mode 100644
index 0000000..af7c06e
--- /dev/null
+++ b/frontend/admin/tsconfig.node.json
@@ -0,0 +1,10 @@
+{
+ "compilerOptions": {
+ "composite": true,
+ "skipLibCheck": true,
+ "module": "ESNext",
+ "moduleResolution": "node",
+ "allowSyntheticDefaultImports": true
+ },
+ "include": ["vite.config.ts"]
+}
diff --git a/frontend/admin/vite.config.ts b/frontend/admin/vite.config.ts
new file mode 100644
index 0000000..79f5b83
--- /dev/null
+++ b/frontend/admin/vite.config.ts
@@ -0,0 +1,24 @@
+import { defineConfig } from 'vite';
+import react from '@vitejs/plugin-react';
+import { resolve } from 'path'
+
+
+// https://vitejs.dev/config/
+export default defineConfig({
+ plugins: [react()],
+ define: {
+ 'process.env': process.env,
+ },
+ server: {
+ host: true,
+ port: parseInt(process.env.FRONTEND_ADMIN_PORT),
+ strictPort: true
+ },
+ base: '/admin',
+ esbuild: {
+ keepNames: true,
+ },
+ build: {
+ sourcemap: true,
+ },
+});
diff --git a/frontend/docker/nginx.conf b/frontend/docker/nginx.conf
new file mode 100644
index 0000000..b921bbf
--- /dev/null
+++ b/frontend/docker/nginx.conf
@@ -0,0 +1,113 @@
+server {
+ listen 80;
+ server_name ${DOMAIN_NAME};
+ root /app/public;
+
+ access_log /var/log/nginx/access.log;
+ error_log /var/log/nginx/error.log;
+
+
+
+ location ~ ^/api/ {
+ rewrite ^ /index.php last;
+ }
+
+
+ # Symfony backend API
+ location ~ ^/index.php(/|$) {
+ include fastcgi_params;
+
+ fastcgi_pass backend:9000;
+ fastcgi_split_path_info ^(.+\.php)(/.*)$;
+ client_max_body_size 5m;
+ fastcgi_buffers 4 8k;
+ fastcgi_buffer_size 8k;
+ fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
+ fastcgi_param DOCUMENT_ROOT $document_root;
+
+ internal;
+ }
+
+ location ~ \.php$ {
+ return 404;
+ }
+
+ # React client-side
+ location / {
+ # ----------------------------------------------------------------------
+ # | Compression |
+ # ----------------------------------------------------------------------
+ # https://github.com/h5bp/server-configs-nginx/blob/main/h5bp/web_performance/compression.conf
+
+ # https://nginx.org/en/docs/http/ngx_http_gzip_module.html
+
+ # Enable gzip compression.
+ # Default: off
+ gzip on;
+
+ # Compression level (1-9).
+ # 5 is a perfect compromise between size and CPU usage, offering about 75%
+ # reduction for most ASCII files (almost identical to level 9).
+ # Default: 1
+ gzip_comp_level 5;
+
+ # Don't compress anything that's already small and unlikely to shrink much if at
+ # all (the default is 20 bytes, which is bad as that usually leads to larger
+ # files after gzipping).
+ # Default: 20
+ gzip_min_length 256;
+
+ # Compress data even for clients that are connecting to us via proxies,
+ # identified by the "Via" header (required for CloudFront).
+ # Default: off
+ gzip_proxied any;
+
+ # Tell proxies to cache both the gzipped and regular version of a resource
+ # whenever the client's Accept-Encoding capabilities header varies;
+ # Avoids the issue where a non-gzip capable client (which is extremely rare
+ # today) would display gibberish if their proxy gave them the gzipped version.
+ # Default: off
+ gzip_vary on;
+
+ # Compress all output labeled with one of the following MIME-types.
+ # `text/html` is always compressed by gzip module.
+ # Default: text/html
+ gzip_types
+ application/atom+xml
+ application/geo+json
+ application/javascript
+ application/x-javascript
+ application/json
+ application/ld+json
+ application/manifest+json
+ application/rdf+xml
+ application/rss+xml
+ application/vnd.ms-fontobject
+ application/wasm
+ application/x-web-app-manifest+json
+ application/xhtml+xml
+ application/xml
+ font/eot
+ font/otf
+ font/ttf
+ image/bmp
+ image/svg+xml
+ text/cache-manifest
+ text/calendar
+ text/css
+ text/javascript
+ text/markdown
+ text/plain
+ text/xml
+ text/vcard
+ text/vnd.rim.location.xloc
+ text/vtt
+ text/x-component
+ text/x-cross-domain-policy;
+
+
+ try_files $uri /index.html;
+ }
+}
+
+
diff --git a/frontend/docker/nginx.dev.conf b/frontend/docker/nginx.dev.conf
new file mode 100644
index 0000000..90b5112
--- /dev/null
+++ b/frontend/docker/nginx.dev.conf
@@ -0,0 +1,74 @@
+server {
+ listen 80;
+ listen [::]:80;
+ server_name ${DOMAIN_NAME};
+ client_max_body_size 8M;
+
+ root /app/public;
+ index index.php index.html index.htm;
+
+ access_log /var/log/nginx/access.log;
+ error_log /var/log/nginx/error.log;
+
+ location ~ ^/(storage|assets) {
+
+ }
+
+ location ~ ^/(api|resume) {
+ rewrite ^ /index.php last;
+ }
+
+ location ~ ^/_profiler/ {
+ rewrite ^ /index.php last;
+ }
+
+ # Symfony backend API
+ location ~ ^/index.php(/|$) {
+ include fastcgi_params;
+
+ fastcgi_pass backend:9000;
+ fastcgi_split_path_info ^(.+\.php)(/.*)$;
+ client_max_body_size 5m;
+ fastcgi_buffers 4 8k;
+ fastcgi_buffer_size 8k;
+ fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
+ fastcgi_param DOCUMENT_ROOT $document_root;
+
+ internal;
+ }
+
+ location ~ \.php$ {
+ return 404;
+ }
+
+ location /admin/ {
+
+ proxy_http_version 1.1;
+ proxy_cache_bypass $http_upgrade;
+
+ proxy_set_header Upgrade $http_upgrade;
+ proxy_set_header Connection 'upgrade';
+ proxy_set_header Host $host;
+ proxy_set_header X-Real-IP $remote_addr;
+ proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
+ proxy_set_header X-Forwarded-Proto $scheme;
+
+ proxy_pass http://frontend-admin:${FRONTEND_ADMIN_PORT};
+ }
+
+ location / {
+ proxy_http_version 1.1;
+ proxy_cache_bypass $http_upgrade;
+
+ proxy_set_header Upgrade $http_upgrade;
+ proxy_set_header Connection 'upgrade';
+ proxy_set_header Host $host;
+ proxy_set_header X-Real-IP $remote_addr;
+ proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
+ proxy_set_header X-Forwarded-Proto $scheme;
+
+ proxy_pass http://frontend-public:${FRONTEND_PUBLIC_PORT};
+ }
+}
+
+
diff --git a/frontend/public/.env b/frontend/public/.env
new file mode 100644
index 0000000..ffde7aa
--- /dev/null
+++ b/frontend/public/.env
@@ -0,0 +1 @@
+API_URL='http://rashin.me.local/api'
diff --git a/frontend/public/.gitignore b/frontend/public/.gitignore
new file mode 100644
index 0000000..a88f2f8
--- /dev/null
+++ b/frontend/public/.gitignore
@@ -0,0 +1,23 @@
+# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
+
+# dependencies
+/node_modules
+/.pnp
+.pnp.js
+
+# testing
+/coverage
+
+# production
+/frontend/build
+
+# misc
+.DS_Store
+.env.local
+.env.development.local
+.env.test.local
+.env.production.local
+
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
diff --git a/frontend/public/Dockerfile b/frontend/public/Dockerfile
new file mode 100644
index 0000000..37e4a6a
--- /dev/null
+++ b/frontend/public/Dockerfile
@@ -0,0 +1,31 @@
+FROM node:latest AS base
+
+WORKDIR /app
+
+COPY entrypoint.sh /entrypoint.sh
+RUN chmod +x /entrypoint.sh
+
+RUN set -x && npm install -g pnpm
+
+FROM base AS dev
+
+ADD ./ /app
+
+RUN pnpm install
+
+ENTRYPOINT ["/entrypoint.sh"]
+
+CMD ["pnpm", "run", "dev"]
+
+#CMD ["nginx", "-g", "daemon off;"]
+
+
+#FROM base AS prod
+
+#WORKDIR /app/public
+
+#RUN npm install --force && npm run build
+
+#COPY --from=base /frontend/build/ ./
+
+#CMD ["npm", "build"]
diff --git a/frontend/public/entrypoint.sh b/frontend/public/entrypoint.sh
new file mode 100644
index 0000000..adbfe8c
--- /dev/null
+++ b/frontend/public/entrypoint.sh
@@ -0,0 +1,5 @@
+#!/bin/sh
+
+npm rebuild esbuild
+
+exec "$@"
diff --git a/frontend/public/index.html b/frontend/public/index.html
new file mode 100644
index 0000000..7ce864d
--- /dev/null
+++ b/frontend/public/index.html
@@ -0,0 +1,14 @@
+
+
+
+ Sergey Rashin - Portfolio
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/public/package.json b/frontend/public/package.json
new file mode 100644
index 0000000..11f52ec
--- /dev/null
+++ b/frontend/public/package.json
@@ -0,0 +1,57 @@
+{
+ "name": "rashinme-frontend",
+ "version": "0.1.0",
+ "private": true,
+ "type": "module",
+ "dependencies": {
+ "@emotion/react": "^11.10.6",
+ "axios": "^1.3.6",
+ "bootstrap": "^5.2.3",
+ "prop-types": "^15.8.1",
+ "react": "^18.2.0",
+ "react-awesome-reveal": "^4.2.3",
+ "react-bootstrap": "^2.7.4",
+ "react-chrono": "^2.2.0",
+ "react-dark-mode-toggle": "^0.2.0",
+ "react-dark-mode-toggle-2": "^2.0.8",
+ "react-dom": "^18.2.0",
+ "react-markdown": "^8.0.7",
+ "react-router": "^6.10.0",
+ "react-router-dom": "^6.10.0",
+ "react-social-icons": "^5.15.0",
+ "styled-components": "^5.3.10",
+ "typewriter-effect": "^2.19.0",
+ "use-dark-mode": "^2.3.1",
+ "use-dark-mode-ts": "^1.0.2",
+ "vertical-timeline-component-for-react": "^1.0.7",
+ "web-vitals": "^3.3.1"
+ },
+ "devDependencies": {
+ "@testing-library/jest-dom": "^5.16.5",
+ "@testing-library/react": "^14.0.0",
+ "@testing-library/user-event": "^14.4.3",
+ "@types/node": "^20.1.4",
+ "@types/prop-types": "^15.7.5",
+ "@types/react": "^18.2.0",
+ "@types/react-dom": "^18.2.1",
+ "@types/react-router": "^5.1.20",
+ "@types/react-router-dom": "^5.3.3",
+ "@types/styled-components": "^5.1.26",
+ "@vitejs/plugin-react": "^4.0.0",
+ "eslint": "^8.39.0",
+ "eslint-config-airbnb": "^19.0.4",
+ "eslint-plugin-import": "^2.27.5",
+ "eslint-plugin-jsx-a11y": "^6.7.1",
+ "eslint-plugin-react": "^7.32.2",
+ "eslint-plugin-react-hooks": "^4.6.0",
+ "react-error-overlay": "^6.0.11",
+ "source-map-explorer": "^2.5.3",
+ "vite": "^4.3.2"
+ },
+ "scripts": {
+ "analyze": "source-map-explorer 'build/static/js/*.js'",
+ "dev": "vite",
+ "build": "vite build",
+ "preview": "vite preview"
+ }
+}
diff --git a/frontend/public/public/images/about/profile.png b/frontend/public/public/images/about/profile.png
new file mode 100644
index 0000000..e63914a
Binary files /dev/null and b/frontend/public/public/images/about/profile.png differ
diff --git a/frontend/public/public/images/logo.png b/frontend/public/public/images/logo.png
new file mode 100644
index 0000000..96154ba
Binary files /dev/null and b/frontend/public/public/images/logo.png differ
diff --git a/frontend/public/public/images/logo.svg b/frontend/public/public/images/logo.svg
new file mode 100644
index 0000000..6e1f532
--- /dev/null
+++ b/frontend/public/public/images/logo.svg
@@ -0,0 +1,3 @@
+
+
+
diff --git a/frontend/public/public/logo.svg b/frontend/public/public/logo.svg
new file mode 100644
index 0000000..4723244
--- /dev/null
+++ b/frontend/public/public/logo.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/public/public/manifest.json b/frontend/public/public/manifest.json
new file mode 100644
index 0000000..f73c3bf
--- /dev/null
+++ b/frontend/public/public/manifest.json
@@ -0,0 +1,15 @@
+{
+ "short_name": "Sergey Rashin",
+ "name": "Sergey Rashin - Portfolio",
+ "icons": [
+ {
+ "src": "logo.svg",
+ "type": "image/svg",
+ "sizes": "835x519"
+ }
+ ],
+ "start_url": ".",
+ "display": "standalone",
+ "theme_color": "#000000",
+ "background_color": "#ffffff"
+}
diff --git a/frontend/public/public/robots.txt b/frontend/public/public/robots.txt
new file mode 100644
index 0000000..e9e57dc
--- /dev/null
+++ b/frontend/public/public/robots.txt
@@ -0,0 +1,3 @@
+# https://www.robotstxt.org/robotstxt.html
+User-agent: *
+Disallow:
diff --git a/frontend/public/src/App.css b/frontend/public/src/App.css
new file mode 100644
index 0000000..2dd34e3
--- /dev/null
+++ b/frontend/public/src/App.css
@@ -0,0 +1,180 @@
+html, body {
+ margin:0;
+ height:100%;
+ width: 100%;
+ display: block;
+}
+
+#root {
+ width: 100%;
+ height: 100%;
+ margin:0;
+ display: flex;
+ flex-direction: column;
+}
+
+
+nav.navbar {
+ flex: 0 0 0;
+ background:red;
+}
+
+.main {
+ flex: auto;
+ overflow-y: auto;
+ flex-direction: column;
+ text-align: center;
+ display: block;
+ align-items: center;
+ justify-content: center;
+}
+
+.main > div {
+ display: inline;
+}
+
+.main .header {
+ display: inline-block;
+ margin-top: 12px;
+}
+
+
+.navbar-custom {
+ font-size:large;
+}
+
+.navbar__link {
+ margin-left: 0.75em;
+ margin-right: 0.75em;
+ font-size: 1em;
+ cursor: pointer;
+ text-decoration: none;
+ letter-spacing: .1em;
+ text-indent: .3em;
+ border-bottom: 3px solid transparent;
+}
+
+.navbar__link::after {
+ transition: all ease-in-out .2s;
+ background: none repeat scroll 0 0;
+ content: "";
+ display: block;
+ margin-top: 2px;
+ height: 3px;
+ width: 0;
+}
+
+ .navbar__link:hover::after {
+ visibility: visible;
+ width: 40%;
+}
+
+.navbar__link.active::after {
+ transition: all ease-in-out .2s;
+ width: 100%;
+}
+
+.navbar__link.active:hover::after {
+ width: 100%;
+}
+
+/*.MainApp {*/
+/* margin-top: 8vh;*/
+/* height: 90vh;*/
+/*}*/
+
+/*.main {*/
+/* display: flex;*/
+/* flex-direction: column;*/
+/* height: 100%;*/
+/*}*/
+
+.section-content-container {
+ display: flex;
+ flex-direction: column;
+ flex-grow: 1;
+ justify-content: center;
+ align-items: center;
+}
+
+.Typewriter {
+ display: inline-block;
+}
+
+.Typewriter__wrapper {
+ font-size: xx-large;
+ font-weight: bold;
+}
+
+.Typewriter__cursor {
+ font-size: xx-large;
+ font-weight: bold;
+}
+
+section.timeline-card-content {
+ justify-content: center;
+}
+
+section.timeline-card-content p.card-title {
+ font-size: 1.5em;
+ margin-top: 25px;
+}
+
+section.timeline-card-content p.card-sub-title {
+ font-size: 1.25em;
+ margin-top: 10px;
+}
+
+section.timeline-card-content div.card-description > p {
+ font-size: 1.1em;
+ margin-top: 10px;
+ font-weight: 500;
+}
+
+.social {
+ position: relative;
+ margin-top: 60px;
+}
+
+
+@media only screen and (min-width: 768px) and (max-width: 992px) {
+
+ .navbar__link {
+ margin-left: 0.4em;
+ margin-right: 0.4em;
+ font-size: 0.9em;
+ cursor: pointer;
+ letter-spacing: .1em;
+ text-indent: .2em;
+ }
+}
+
+@media (max-width: 768px) {
+
+ .navbar__link::after {
+ background: none;
+ content: "";
+ display: none;
+ margin-top: 0;
+ height: 0;
+ width: 0;
+ }
+
+ .navbar__link--active::after {
+ width: 0;
+ }
+
+ .navbar__link--active:hover::after {
+ width: 0;
+ }
+
+ .navbar__link:hover::after {
+ width: 0;
+ }
+}
+
+.header {
+ font-size: 3em;
+ font-weight: bold;
+ margin-bottom: 25px;
+}
diff --git a/frontend/public/src/App.test.ts b/frontend/public/src/App.test.ts
new file mode 100644
index 0000000..1f03afe
--- /dev/null
+++ b/frontend/public/src/App.test.ts
@@ -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/frontend/public/src/App.tsx b/frontend/public/src/App.tsx
new file mode 100644
index 0000000..0b11f3c
--- /dev/null
+++ b/frontend/public/src/App.tsx
@@ -0,0 +1,28 @@
+import './App.css';
+import 'bootstrap/dist/css/bootstrap.min.css';
+import {BrowserRouter} from 'react-router-dom';
+import { ThemeProvider } from 'styled-components';
+import useDarkMode from 'use-dark-mode';
+import {AppContextProvider} from './AppContext';
+import MainApp from './MainApp';
+import GlobalStyles from './theme/GlobalStyles';
+import { lightTheme, darkTheme } from './theme/themes';
+import NavBarWithRouter from "./components/NavBar";
+
+function App() {
+ const darkMode = useDarkMode();
+
+ return (
+
+
+
+
+
+
+
+
+
+ );
+}
+
+export default App;
diff --git a/frontend/public/src/AppContext.ts b/frontend/public/src/AppContext.ts
new file mode 100644
index 0000000..e20364a
--- /dev/null
+++ b/frontend/public/src/AppContext.ts
@@ -0,0 +1,12 @@
+import * as React from 'react';
+import {DarkMode} from "use-dark-mode";
+
+export interface AppContextInterface {
+ darkMode: DarkMode
+}
+
+const {Provider, Consumer} = React.createContext(null);
+
+export const AppContextProvider = Provider;
+
+export const AppContextConsumer = Consumer;
diff --git a/frontend/public/src/MainApp.tsx b/frontend/public/src/MainApp.tsx
new file mode 100644
index 0000000..a5c58fa
--- /dev/null
+++ b/frontend/public/src/MainApp.tsx
@@ -0,0 +1,32 @@
+import { Suspense } from 'react';
+import { Routes, Route } from 'react-router-dom';
+import FallbackSpinner from './components/FallbackSpinner';
+import NotFound from "./components/NotFound";
+import ExperiencePage from "./pages/Experience/ExperiencePage";
+import HomePage from "./pages/Home/HomePage";
+import AboutPage from "./pages/About/AboutPage";
+import EducationPage from "./pages/Education/EducationPage";
+import ProjectsPage from "./pages/Project/ProjectsPage";
+import SkillPage from "./pages/Skill/SkillPage";
+
+function MainApp() {
+ return (
+
+ }>
+
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+
+
+
+ );
+}
+
+export default MainApp;
+
+
diff --git a/frontend/public/src/components/FallbackSpinner.tsx b/frontend/public/src/components/FallbackSpinner.tsx
new file mode 100644
index 0000000..17329f2
--- /dev/null
+++ b/frontend/public/src/components/FallbackSpinner.tsx
@@ -0,0 +1,21 @@
+import { Spinner } from 'react-bootstrap';
+import {CSSProperties} from "react";
+
+const styles: {[key: string]: CSSProperties} = {
+ spinnerContainerStyle: {
+ display: 'flex',
+ justifyContent: 'center',
+ alignItems: 'center',
+ height: '100%',
+ },
+};
+
+const FallbackSpinner = () => {
+ return (
+
+
+
+ );
+}
+
+export default FallbackSpinner;
diff --git a/frontend/public/src/components/Header.tsx b/frontend/public/src/components/Header.tsx
new file mode 100644
index 0000000..c412c71
--- /dev/null
+++ b/frontend/public/src/components/Header.tsx
@@ -0,0 +1,11 @@
+import '../App.css';
+
+interface HeaderProps {
+ title: string,
+}
+
+const Header = ({ title }: HeaderProps) => {
+ return {title}
;
+}
+
+export default Header;
diff --git a/frontend/public/src/components/NavBar.tsx b/frontend/public/src/components/NavBar.tsx
new file mode 100644
index 0000000..ec6eb9d
--- /dev/null
+++ b/frontend/public/src/components/NavBar.tsx
@@ -0,0 +1,106 @@
+import { Navbar, Nav, Container } from 'react-bootstrap';
+import {useState, useContext, CSSProperties} from 'react';
+import { NavLink } from 'react-router-dom';
+import styled, { ThemeContext } from 'styled-components';
+import ThemeToggler from './ThemeToggler';
+import navbar from "../constants/navbar";
+
+const styles: {[key: string]: CSSProperties} = {
+ logoStyle: {
+ width: 50,
+ height: 40,
+ },
+};
+
+const ExternalNavLink = styled.a`
+ color: ${(props) => props.theme.navbarTheme.linkColor};
+ &:hover {
+ color: ${(props) => props.theme.navbarTheme.linkHoverColor};
+ }
+ &::after {
+ background-color: ${(props) => props.theme.accentColor};
+ }
+`;
+
+const InternalNavLink = styled(NavLink)`
+ color: ${(props) => props.theme.navbarTheme.linkColor};
+ &:hover {
+ color: ${(props) => props.theme.navbarTheme.linkHoverColor};
+ }
+ &::after {
+ background-color: ${(props) => props.theme.accentColor};
+ }
+ &.navbar__link.active {
+ color: ${(props) => props.theme.navbarTheme.linkActiveColor};
+ }
+`;
+
+const NavBar = () => {
+ const theme = useContext(ThemeContext);
+ const [expanded, setExpanded] = useState(false);
+
+ return (
+
+
+ {navbar?.logo && (
+
+
+
+ )}
+ setExpanded(!expanded)}
+ />
+
+
+
+ setExpanded(false)}
+ />
+
+
+
+ );
+};
+
+export default NavBar;
diff --git a/frontend/public/src/components/NotFound.tsx b/frontend/public/src/components/NotFound.tsx
new file mode 100644
index 0000000..0df0a87
--- /dev/null
+++ b/frontend/public/src/components/NotFound.tsx
@@ -0,0 +1,24 @@
+import { Link } from "react-router-dom";
+import {CSSProperties} from "react";
+
+const styles: {[key: string]: CSSProperties} = {
+ mainContainer: {
+ height: '100%',
+ display: 'flex',
+ flexDirection: 'column',
+ justifyContent: 'center',
+ alignItems: 'center',
+ }
+};
+
+function NotFound() {
+ return (
+
+
404 - Not Found!
+
+ Go to the home page
+
+
+ );
+}
+export default NotFound;
diff --git a/frontend/public/src/components/Social.tsx b/frontend/public/src/components/Social.tsx
new file mode 100644
index 0000000..dc5dd69
--- /dev/null
+++ b/frontend/public/src/components/Social.tsx
@@ -0,0 +1,62 @@
+import {useContext, CSSProperties} from 'react';
+import { SocialIcon } from 'react-social-icons';
+import { ThemeContext } from 'styled-components';
+
+const styles: {[key: string]: CSSProperties} = {
+ iconStyle: {
+ marginLeft: 10,
+ marginRight: 10,
+ marginBottom: 10,
+ },
+};
+
+interface SocialParams {
+ github: string,
+ linkedin: string,
+ email: string,
+}
+
+const Social = ({github, linkedin, email}: SocialParams) => {
+ const theme = useContext(ThemeContext);
+ const mailTo = 'mailto:' + email;
+
+ return (
+
+
+ {linkedin ? (
+
+ ) : null}
+
+ {github ? (
+
+ ) : null}
+
+ {github ? (
+
+ ) : null}
+
+ );
+}
+
+export default Social;
diff --git a/frontend/public/src/components/ThemeToggler.tsx b/frontend/public/src/components/ThemeToggler.tsx
new file mode 100644
index 0000000..c47b8ab
--- /dev/null
+++ b/frontend/public/src/components/ThemeToggler.tsx
@@ -0,0 +1,30 @@
+import DarkModeToggle from 'react-dark-mode-toggle';
+import {AppContextConsumer} from '../AppContext';
+import {DarkMode} from "use-dark-mode";
+
+export interface ThemeTogglerInterface {
+ onClick: () => void
+}
+
+const ThemeToggler = ({onClick}: ThemeTogglerInterface) => {
+ const handleOnChange = (darkMode: DarkMode) => {
+ darkMode.toggle();
+ onClick();
+ };
+
+ return (
+
+ {appContext => appContext && (
+
+ handleOnChange(appContext.darkMode)}
+ checked={appContext.darkMode.value}
+ size={50}
+ />
+
+ )}
+
+ );
+}
+
+export default ThemeToggler;
diff --git a/frontend/public/src/constants/endpoints.ts b/frontend/public/src/constants/endpoints.ts
new file mode 100644
index 0000000..fe73bf5
--- /dev/null
+++ b/frontend/public/src/constants/endpoints.ts
@@ -0,0 +1,13 @@
+const endpoints = {
+ navbar: 'profile/navbar.json',
+ routes: 'profile/routes.json',
+ home: 'profile/home.json',
+ social: 'profile/social.json',
+ about: 'profile/about.json',
+ skills: 'profile/skills.json',
+ education: 'profile/education.json',
+ experiences: 'profile/experiences.json',
+ projects: 'profile/projects.json',
+};
+
+export default endpoints;
diff --git a/frontend/public/src/constants/navbar.ts b/frontend/public/src/constants/navbar.ts
new file mode 100644
index 0000000..c056f80
--- /dev/null
+++ b/frontend/public/src/constants/navbar.ts
@@ -0,0 +1,40 @@
+const navbar = {
+ logo : {
+ source: "images/logo.png",
+ height : 45,
+ width : 50
+ },
+ sections: [
+ {
+ title: "Home",
+ href: "/",
+ },
+ {
+ title: "About",
+ href: "/about"
+ },
+ {
+ title: "Skills",
+ href: "/skills"
+ },
+ {
+ title: "Education",
+ href: "/education"
+ },
+ {
+ title: "Experience",
+ href: "/experience"
+ },
+ {
+ title: "Projects",
+ href: "/projects"
+ },
+ {
+ title: "Resume",
+ href: "/resume",
+ type: "link"
+ }
+ ]
+};
+
+export default navbar;
diff --git a/frontend/public/src/constants/routes.ts b/frontend/public/src/constants/routes.ts
new file mode 100644
index 0000000..7ce7d8f
--- /dev/null
+++ b/frontend/public/src/constants/routes.ts
@@ -0,0 +1,29 @@
+const routes = [
+ {
+ component: "About",
+ path: "/about",
+ headerTitle: "About"
+ },
+ {
+ component: "Skills",
+ path: "/skills",
+ headerTitle: "Skills"
+ },
+ {
+ component: "Education",
+ path: "/education",
+ headerTitle: "Education"
+ },
+ {
+ component: "Experience",
+ path: "/experience",
+ headerTitle: "Experience"
+ },
+ {
+ component: "Projects",
+ path: "/projects",
+ headerTitle: "Projects"
+ }
+];
+
+export default routes;
diff --git a/frontend/public/src/css/education.css b/frontend/public/src/css/education.css
new file mode 100644
index 0000000..50cb971
--- /dev/null
+++ b/frontend/public/src/css/education.css
@@ -0,0 +1,19 @@
+.vertical {
+ justify-content: center;
+}
+
+.vertical_alternating {
+ justify-content: center;
+}
+
+@media only screen and (max-width: 578px) {
+ div.vertical-item-row > div:nth-child(1) {
+ width: 20% !important;
+ }
+ div.vertical-item-row > div:nth-child(2) {
+ width: 65% !important;
+ }
+ div.vertical-item-row > div:nth-child(3) {
+ width: 15% !important;
+ }
+}
diff --git a/frontend/public/src/css/experience.css b/frontend/public/src/css/experience.css
new file mode 100644
index 0000000..e31c2c9
--- /dev/null
+++ b/frontend/public/src/css/experience.css
@@ -0,0 +1,9 @@
+.entry .body{
+ margin: 0 0 1em;
+}
+
+@media only screen and (max-width: 768px) {
+ .item-title {
+ margin-top: 10px !important;
+ }
+}
diff --git a/frontend/public/src/index.css b/frontend/public/src/index.css
new file mode 100644
index 0000000..ec2585e
--- /dev/null
+++ b/frontend/public/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/frontend/public/src/index.tsx b/frontend/public/src/index.tsx
new file mode 100644
index 0000000..1c3e3c3
--- /dev/null
+++ b/frontend/public/src/index.tsx
@@ -0,0 +1,14 @@
+import * as ReactDOMClient from 'react-dom/client';
+import App from './App';
+import reportWebVitals from './reportWebVitals';
+
+const container = document.getElementById('root')!;
+
+// Create a root.
+const root = ReactDOMClient.createRoot(container);
+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/frontend/public/src/logo.svg b/frontend/public/src/logo.svg
new file mode 100644
index 0000000..9dfc1c0
--- /dev/null
+++ b/frontend/public/src/logo.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/public/src/pages/About/AboutPage.tsx b/frontend/public/src/pages/About/AboutPage.tsx
new file mode 100644
index 0000000..ff390ac
--- /dev/null
+++ b/frontend/public/src/pages/About/AboutPage.tsx
@@ -0,0 +1,96 @@
+import {useState, useEffect, CSSProperties} from 'react';
+import ReactMarkdown from 'react-markdown';
+import { Container, Col, Row } from 'react-bootstrap';
+import { Fade } from 'react-awesome-reveal';
+import ApiService, {Property} from "../../services/ApiService";
+import FallbackSpinner from "../../components/FallbackSpinner";
+import Header from "../../components/Header";
+
+const styles: {[key: string]: CSSProperties} = {
+ introTextContainer: {
+ margin: 10,
+ flexDirection: 'column',
+ whiteSpace: 'pre-wrap',
+ textAlign: 'left',
+ fontSize: '1.2em',
+ fontWeight: 500,
+ },
+ introImageContainer: {
+ margin: 10,
+ justifyContent: 'center',
+ alignItems: 'center',
+ display: 'flex',
+ },
+ imgStyle: {
+ maxHeight: '240px',
+ maxWidth: '240px',
+ }
+};
+
+interface AboutProps {
+ header: string,
+}
+
+const About = ({ header }: AboutProps) => {
+ const [userPhoto, setUserPhoto] = useState(null);
+ const [about, setAbout] = useState('');
+ const [isLoading, setIsLoading] = useState(true);
+
+ const parseIntro = (text: string) => (
+
+ );
+
+ useEffect(() => {
+ ApiService.getConfiguration([
+ 'userPhoto',
+ 'about',
+ ])
+ .then((properties: Property[]) => {
+ properties.map((property: Property) => {
+ switch (property.key) {
+ case 'userPhoto':
+ ApiService.getFile(parseInt(property.value))
+ .then((file) => {
+ setUserPhoto(file.path + file.name);
+ })
+ break;
+ case 'about':
+ setAbout(property.value)
+ break;
+ }
+ setIsLoading(false)
+ })
+ });
+ }, []);
+
+ return (
+ <>
+
+
+
+ {isLoading === false
+ ? (
+
+
+
+ {parseIntro(about)}
+
+ {userPhoto != null
+ ? (
+
+
+
+ ) : null}
+
+
+ )
+ : }
+
+
+ >
+ );
+}
+
+export default About;
diff --git a/frontend/public/src/pages/Education/EducationPage.tsx b/frontend/public/src/pages/Education/EducationPage.tsx
new file mode 100644
index 0000000..fb7bf5a
--- /dev/null
+++ b/frontend/public/src/pages/Education/EducationPage.tsx
@@ -0,0 +1,71 @@
+import { useEffect, useState, useContext } from 'react';
+import { Chrono } from 'react-chrono';
+import { Container } from 'react-bootstrap';
+import { Fade } from 'react-awesome-reveal';
+import { ThemeContext } from 'styled-components';
+import '../../css/education.css';
+import FallbackSpinner from "../../components/FallbackSpinner";
+import ApiService, {PaginatedCollection, Education} from "../../services/ApiService";
+import Header from "../../components/Header";
+import {TimelineItemModel} from "react-chrono/dist/models/TimelineItemModel";
+
+interface EducationProps {
+ header: string,
+}
+
+const EducationPage = ({ header }: EducationProps) => {
+ const theme = useContext(ThemeContext);
+ const [data, setData] = useState(null);
+
+ const loadData = () => {
+ ApiService.getEducations()
+ .then((response: PaginatedCollection) => {
+ let items = response.data.map((item: Education) => {
+ return {
+ title: `${item.from.month}/${item.from.year} - ${item.to ? item.to.month + '/' + item.to.year : 'Until Now'}`,
+ cardTitle: item.institution,
+ cardSubtitle: item.faculty,
+ cardDetailedText: item.specialization,
+ }
+ })
+ setData(items);
+ });
+ };
+
+ useEffect(() => {
+ loadData();
+ }, []);
+
+ return (
+ <>
+
+ {data ? (
+
+
+
+
+
+
+
+
+ ) : }
+ >
+ );
+}
+
+export default EducationPage;
diff --git a/frontend/public/src/pages/Experience/ExperiencePage.tsx b/frontend/public/src/pages/Experience/ExperiencePage.tsx
new file mode 100644
index 0000000..f3efbd7
--- /dev/null
+++ b/frontend/public/src/pages/Experience/ExperiencePage.tsx
@@ -0,0 +1,106 @@
+import {useEffect, useState, useContext, CSSProperties} from 'react';
+// @ts-ignore
+import { Timeline, TimelineItem } from 'vertical-timeline-component-for-react';
+import { Container } from 'react-bootstrap';
+import ReactMarkdown from 'react-markdown';
+import { ThemeContext } from 'styled-components';
+import { Fade } from 'react-awesome-reveal';
+import '../../css/experience.css';
+import ApiService, {Job, PaginatedCollection} from "../../services/ApiService";
+import Header from "../../components/Header";
+import FallbackSpinner from "../../components/FallbackSpinner";
+
+const styles: {[key: string]: CSSProperties} = {
+ ulStyle: {
+ listStylePosition: 'outside',
+ paddingLeft: 20,
+ },
+ subtitleContainerStyle: {
+ marginTop: 10,
+ marginBottom: 10,
+ },
+ subtitleStyle: {
+ display: 'inline-block',
+ },
+ inlineChild: {
+ display: 'inline-block',
+ },
+ itemStyle: {
+ marginBottom: 10,
+ },
+};
+
+interface ExperienceListProps {
+ header: string
+}
+
+const ExperiencePage = ({ header }: ExperienceListProps) => {
+ const theme = useContext(ThemeContext);
+
+ const [data, setData] = useState | null>(null);
+
+ useEffect(() => {
+ ApiService.getJobs()
+ .then((response: PaginatedCollection) => {
+ setData(response)
+ });
+ }, []);
+
+ return (
+ <>
+
+
+ {data
+ ? (
+
+
+
+
+ {data.data.map((item: Job) => (
+
+
+ {item.name}
+
+
+
+ {item.type && (
+
+ ·
+ {' '}
+ {item.type}
+
+ )}
+
+
+
+ ))}
+
+
+
+
+ ) : }
+ >
+ );
+}
+
+export default ExperiencePage;
diff --git a/frontend/public/src/pages/Home/HomePage.tsx b/frontend/public/src/pages/Home/HomePage.tsx
new file mode 100644
index 0000000..5dcde93
--- /dev/null
+++ b/frontend/public/src/pages/Home/HomePage.tsx
@@ -0,0 +1,88 @@
+import {useState, useEffect, CSSProperties} from 'react';
+import Typewriter from 'typewriter-effect';
+import { Fade } from 'react-awesome-reveal';
+import ApiService, {Property} from "../../services/ApiService";
+import Social from "../../components/Social";
+import FallbackSpinner from "../../components/FallbackSpinner";
+
+const styles: {[key: string]: CSSProperties} = {
+ nameStyle: {
+ fontSize: '5em',
+ },
+ inlineChild: {
+ display: 'inline-block',
+ },
+ mainContainer: {
+ height: '100%',
+ display: 'flex',
+ flexDirection: 'column',
+ justifyContent: 'center',
+ alignItems: 'center',
+ },
+};
+
+const HomePage = () => {
+ const [userName, setUserName] = useState('');
+ const [roles, setRoles] = useState([]);
+ const [githubLink, setGithubLink] = useState('');
+ const [linkedinLink, setLinkedinLink] = useState('');
+ const [emailAddress, setEmailAddress] = useState('');
+ const [isLoading, setIsLoading] = useState(true);
+
+ useEffect(() => {
+ ApiService.getConfiguration([
+ 'userName',
+ 'roles',
+ 'githubLink',
+ 'linkedinLink',
+ 'emailAddress',
+ ])
+ .then((properties: Property[]) => {
+ properties.map((property: Property) => {
+ switch (property.key) {
+ case 'userName':
+ setUserName(property.value)
+ break;
+ case 'roles':
+ setRoles(JSON.parse(property.value))
+ break;
+ case 'githubLink':
+ setGithubLink(property.value)
+ break;
+ case 'linkedinLink':
+ setLinkedinLink(property.value)
+ break;
+ case 'emailAddress':
+ setEmailAddress(property.value)
+ break;
+ }
+ setIsLoading(false)
+ })
+ });
+ }, []);
+
+ return isLoading == false ? (
+
+
+
{userName}
+
+
I'm
+
+
+
+
+
+ ) : ;
+}
+
+export default HomePage;
diff --git a/frontend/public/src/pages/Project/ProjectCard.tsx b/frontend/public/src/pages/Project/ProjectCard.tsx
new file mode 100644
index 0000000..1c18b87
--- /dev/null
+++ b/frontend/public/src/pages/Project/ProjectCard.tsx
@@ -0,0 +1,102 @@
+import {CSSProperties, useContext} from 'react';
+import {
+ Button, Card, Badge, Col,
+} from 'react-bootstrap';
+import { ThemeContext } from 'styled-components';
+import ReactMarkdown from 'react-markdown';
+import {Link, Project} from "../../services/ApiService";
+
+const styles: {[key: string]: CSSProperties} = {
+ badgeStyle: {
+ paddingLeft: 10,
+ paddingRight: 10,
+ paddingTop: 5,
+ paddingBottom: 5,
+ margin: 5,
+ },
+ cardStyle: {
+ borderRadius: 10,
+ },
+ cardTitleStyle: {
+ fontSize: 24,
+ fontWeight: 700,
+ },
+ cardTextStyle: {
+ textAlign: 'left',
+ marginBottom: '-1rem',
+ padding:0,
+ },
+ linkStyle: {
+ textDecoration: 'none',
+ padding: 10,
+ },
+ buttonStyle: {
+ margin: 5,
+ },
+};
+
+interface ProjectProps {
+ project: Project
+}
+
+
+const ProjectCard = ({ project }: ProjectProps) => {
+ const theme = useContext(ThemeContext);
+
+ return (
+
+
+
+
+ {project.name}
+
+
+
+
+
+
+ {project.links?.map((link: Link) => (
+
+ ))}
+
+ {project.tags && (
+
+ {project.tags.map((tag: string) => (
+
+ {tag}
+
+ ))}
+
+ )}
+
+
+ );
+};
+
+export default ProjectCard;
diff --git a/frontend/public/src/pages/Project/ProjectsPage.tsx b/frontend/public/src/pages/Project/ProjectsPage.tsx
new file mode 100644
index 0000000..f564900
--- /dev/null
+++ b/frontend/public/src/pages/Project/ProjectsPage.tsx
@@ -0,0 +1,81 @@
+import {useState, useEffect, useContext, CSSProperties} from 'react';
+import { Container, Row, Button } from 'react-bootstrap';
+import { ThemeContext } from 'styled-components';
+import {Fade} from 'react-awesome-reveal';
+import Header from "../../components/Header";
+import FallbackSpinner from "../../components/FallbackSpinner";
+import ApiService, {PaginatedCollection, Project} from "../../services/ApiService";
+import ProjectCard from "./ProjectCard";
+
+const styles: {[key: string]: CSSProperties} = {
+ containerStyle: {
+ marginBottom: 25,
+ },
+ showMoreStyle: {
+ margin: 25,
+ },
+};
+
+interface ProjectsProps {
+ header: string,
+}
+
+const ProjectsPage = ({ header }: ProjectsProps) => {
+ const theme = useContext(ThemeContext);
+ const [showMore, setShowMore] = useState(false);
+ const [offset, setOffset] = useState(0);
+ const [limit, setLimit] = useState(6);
+ const [data, setData] = useState | null>(null);
+
+ const loadData = () => {
+ ApiService.getProjects(limit, offset)
+ .then((response: PaginatedCollection) => {
+ setData(response);
+
+ const pagination = response.pagination;
+
+ setOffset(pagination.offset);
+ setLimit(pagination.limit);
+
+ setShowMore(pagination.offset + response.data.length < pagination.total);
+ });
+ };
+
+ useEffect( () => {
+ loadData();
+ }, [offset, limit, showMore]);
+
+
+ return (
+ <>
+
+ {data
+ ? (
+
+
+
+ {data.data?.map((project: Project) => (
+
+
+
+ ))}
+
+
+ {showMore
+ && (
+
+ )}
+
+
+ ) : }
+ >
+);
+};
+
+export default ProjectsPage;
diff --git a/frontend/public/src/pages/Skill/SkillPage.tsx b/frontend/public/src/pages/Skill/SkillPage.tsx
new file mode 100644
index 0000000..44c6fa7
--- /dev/null
+++ b/frontend/public/src/pages/Skill/SkillPage.tsx
@@ -0,0 +1,81 @@
+import {CSSProperties, useEffect, useState} from 'react';
+import ReactMarkdown from 'react-markdown';
+import PropTypes from 'prop-types';
+import { Fade } from 'react-awesome-reveal';
+import { Container } from 'react-bootstrap';
+import ApiService, {PaginatedCollection, Section, Skill} from "../../services/ApiService";
+import Header from "../../components/Header";
+import FallbackSpinner from "../../components/FallbackSpinner";
+
+const headerText = 'I love to learn new things and experiment with new technologies.' +
+ 'These are some of the major languages, technologies, tools and platforms I have worked with:';
+
+const styles: {[key: string]: CSSProperties} = {
+ iconStyle: {
+ height: 75,
+ width: 75,
+ margin: 10,
+ marginBottom: 0,
+ },
+ introTextContainer: {
+ whiteSpace: 'pre-wrap',
+ },
+};
+
+interface SkillsProps {
+ header: string,
+}
+
+function Skills({ header }: SkillsProps) {
+ const [data, setData] = useState | null>(null);
+
+ useEffect(() => {
+ ApiService.getSections()
+ .then((response: PaginatedCollection) => {
+ setData(response)
+ });
+ }, []);
+
+ const renderSkillsIntro = (intro: string) => (
+
+
+
+ );
+
+ return (
+ <>
+
+ {data ? (
+
+
+
+ {renderSkillsIntro(headerText)}
+ {data.data?.map((section: Section) => (
+
+
+
{section.name}
+ {section.skills.map((skill: Skill) => (
+
+
+
{skill.name}
+
+ ))}
+
+ ))}
+
+
+
+ ) : }
+ >
+ );
+}
+
+Skills.propTypes = {
+ header: PropTypes.string.isRequired,
+};
+
+export default Skills;
diff --git a/frontend/public/src/reportWebVitals.ts b/frontend/public/src/reportWebVitals.ts
new file mode 100644
index 0000000..d95c0a6
--- /dev/null
+++ b/frontend/public/src/reportWebVitals.ts
@@ -0,0 +1,15 @@
+import { ReportHandler } from 'web-vitals';
+
+const reportWebVitals = (onPerfEntry?: ReportHandler) => {
+ 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/frontend/public/src/services/ApiService.ts b/frontend/public/src/services/ApiService.ts
new file mode 100644
index 0000000..f5e447b
--- /dev/null
+++ b/frontend/public/src/services/ApiService.ts
@@ -0,0 +1,172 @@
+const apiUrl = `${process.env.API_URL}`;
+
+const endpoints = {
+ projects: '/projects',
+ skillSections: '/sections',
+ skills: '/skills',
+ jobs: '/jobs',
+ educations: '/educations',
+ properties: '/properties',
+ storage: '/storage',
+};
+
+export interface Image {
+ readonly id: number,
+ readonly src: string,
+}
+
+export interface File {
+ readonly id: number,
+ readonly name: string,
+ readonly path: string,
+ readonly mimeType: string,
+ readonly size: number,
+}
+
+export interface Link {
+ readonly id: number,
+ readonly title: string,
+ readonly url: string,
+}
+
+export interface Company {
+ readonly name: string,
+ readonly url: string,
+}
+
+export interface WorkDate {
+ readonly month: number,
+ readonly year: number,
+}
+
+export interface Project {
+ readonly id: number,
+ readonly name: string,
+ readonly image: Image,
+ readonly description: string,
+ readonly links: Link[],
+ readonly tags: string[]
+}
+
+
+export interface Job {
+ readonly id: number,
+ readonly image: Image,
+ readonly name: string,
+ readonly type: string,
+ readonly description: string,
+ readonly company: Company,
+ readonly from: WorkDate,
+ readonly to: WorkDate|null,
+}
+
+export interface Education {
+ readonly id: number,
+ readonly institution: string,
+ readonly faculty: string,
+ readonly specialization: string,
+ readonly from: WorkDate,
+ readonly to: WorkDate|null,
+}
+
+export interface Section {
+ readonly id: number,
+ readonly name: string,
+ readonly skills: Skill[],
+}
+
+export interface Skill {
+ readonly id: number,
+ readonly image: Image,
+ readonly name: string,
+ readonly sectionId: number,
+ readonly description: string,
+}
+
+export interface Property {
+ key: string,
+ value: string,
+}
+
+export interface PaginatedCollection {
+ readonly data: T[],
+ readonly pagination: {
+ readonly limit: number,
+ readonly offset: number,
+ readonly total: number
+ }
+}
+
+class ApiService {
+ apiEndpoint: string;
+ constructor(apiEndpoint: string) {
+ this.apiEndpoint = apiEndpoint;
+ }
+
+ async getSections(): Promise>
+ {
+ const url = apiUrl + endpoints.skillSections;
+
+ return await fetch(url, {
+ method: 'GET',
+ })
+ .then((res: Response) => res.json())
+ }
+
+ async getProjects(limit: number, offset: number): Promise>
+ {
+ const url = apiUrl + endpoints.projects + `?limit=${ limit }&offset=${ offset }`;
+
+ return await fetch(url, {
+ method: 'GET',
+ })
+ .then((res: Response) => res.json())
+ }
+
+ async getJobs(): Promise>
+ {
+ const url = apiUrl + endpoints.jobs + '?sort={"field":"from","order":"desc"}';
+
+ return await fetch(url, {
+ method: 'GET',
+ })
+ .then((res: Response) => res.json())
+ }
+
+ async getConfiguration(fields: string[]): Promise
+ {
+ const filterData = {
+ fields: fields,
+ };
+ const url = apiUrl + endpoints.properties + `?filter=${JSON.stringify(filterData)}`;
+
+ return await fetch(url, {
+ method: 'GET',
+ })
+ .then((res: Response) => res.json())
+ }
+
+ async getEducations(): Promise>
+ {
+ const url = apiUrl + endpoints.educations + '?sort={"field":"from","order":"asc"}';
+
+ return await fetch(url, {
+ method: 'GET',
+ })
+ .then((res: Response) => res.json())
+ }
+
+ async getFile(fileId: number): Promise
+ {
+ const url = apiUrl + endpoints.storage + '/' + fileId;
+
+ return await fetch(url, {
+ method: 'GET',
+ })
+ .then((res: Response) => res.json())
+ }
+}
+
+const apiService = new ApiService(apiUrl);
+
+export default apiService;
diff --git a/frontend/public/src/setupTests.js b/frontend/public/src/setupTests.js
new file mode 100644
index 0000000..8f2609b
--- /dev/null
+++ b/frontend/public/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/frontend/public/src/theme/GlobalStyles.ts b/frontend/public/src/theme/GlobalStyles.ts
new file mode 100644
index 0000000..132c225
--- /dev/null
+++ b/frontend/public/src/theme/GlobalStyles.ts
@@ -0,0 +1,11 @@
+import { createGlobalStyle } from 'styled-components';
+
+const GlobalStyles = createGlobalStyle`
+ body {
+ background: ${({ theme }) => theme.background};
+ color: ${({ theme }) => theme.color};
+ transition: all 0.50s linear;
+ }
+`;
+
+export default GlobalStyles;
diff --git a/frontend/public/src/theme/themes.ts b/frontend/public/src/theme/themes.ts
new file mode 100644
index 0000000..a4a92a4
--- /dev/null
+++ b/frontend/public/src/theme/themes.ts
@@ -0,0 +1,47 @@
+export const lightTheme = {
+ background: '#fff',
+ color: '#121212',
+ accentColor: '#3D84C6',
+ chronoTheme: {
+ cardBgColor: 'white',
+ cardForeColor: 'black',
+ titleColor: 'black',
+ titleColorActive: 'black',
+ },
+ timelineLineColor: '#ccc',
+ cardBackground: '#fff',
+ cardFooterBackground: '#f7f7f7',
+ cardBorderColor: '#00000020',
+ navbarTheme: {
+ linkColor: '#dedede',
+ linkHoverColor: '#fefefe',
+ linkActiveColor: '#fefefe',
+ },
+ bsPrimaryVariant: 'light',
+ bsSecondaryVariant: 'dark',
+ socialIconBgColor: '#121212',
+};
+
+export const darkTheme = {
+ background: '#121212',
+ color: '#eee',
+ accentColor: '#3D84C6',
+ chronoTheme: {
+ cardBgColor: '#e7e7e7',
+ cardForeColor: 'white',
+ titleColor: 'white',
+ titleColorActive: 'white',
+ },
+ timelineLineColor: '#444',
+ cardBackground: '#060606',
+ cardFooterBackground: '#181818',
+ cardBorderColor: '#ffffff20',
+ navbarTheme: {
+ linkColor: '#dedede',
+ linkHoverColor: '#fefefe',
+ linkActiveColor: '#fefefe',
+ },
+ bsPrimaryVariant: 'dark',
+ bsSecondaryVariant: 'light',
+ socialIconBgColor: '#fefefe',
+};
diff --git a/frontend/public/tsconfig.json b/frontend/public/tsconfig.json
new file mode 100644
index 0000000..30d6ff1
--- /dev/null
+++ b/frontend/public/tsconfig.json
@@ -0,0 +1,25 @@
+{
+ "compilerOptions": {
+ "target": "ES2020",
+ "useDefineForClassFields": true,
+ "lib": ["ES2020", "DOM", "DOM.Iterable"],
+ "module": "ESNext",
+ "skipLibCheck": true,
+
+ /* Bundler mode */
+ "moduleResolution": "bundler",
+ "allowImportingTsExtensions": true,
+ "resolveJsonModule": true,
+ "isolatedModules": true,
+ "noEmit": true,
+ "jsx": "react-jsx",
+
+ /* Linting */
+ "strict": true,
+ "noUnusedLocals": true,
+ "noUnusedParameters": true,
+ "noFallthroughCasesInSwitch": true
+ },
+ "include": ["src"],
+ "references": [{ "path": "./tsconfig.node.json" }]
+}
diff --git a/frontend/public/tsconfig.node.json b/frontend/public/tsconfig.node.json
new file mode 100644
index 0000000..26063d8
--- /dev/null
+++ b/frontend/public/tsconfig.node.json
@@ -0,0 +1,10 @@
+{
+ "compilerOptions": {
+ "composite": true,
+ "skipLibCheck": true,
+ "module": "ESNext",
+ "moduleResolution": "bundler",
+ "allowSyntheticDefaultImports": true
+ },
+ "include": ["vite.config.ts"]
+}
diff --git a/frontend/public/vite.config.ts b/frontend/public/vite.config.ts
new file mode 100644
index 0000000..1afccd1
--- /dev/null
+++ b/frontend/public/vite.config.ts
@@ -0,0 +1,21 @@
+import { defineConfig } from 'vite';
+import react from '@vitejs/plugin-react'
+
+export default defineConfig({
+ plugins: [react()],
+ define: {
+ 'process.env': process.env,
+ global: {},
+ },
+ server: {
+ host: true,
+ port: parseInt(process.env.FRONTEND_PUBLIC_PORT),
+ strictPort: true
+ },
+ esbuild: {
+ keepNames: true,
+ },
+ build: {
+ sourcemap: true,
+ },
+});
diff --git a/migrations/.gitignore b/migrations/.gitignore
new file mode 100644
index 0000000..e69de29
diff --git a/migrations/Version20230524205355.php b/migrations/Version20230524205355.php
new file mode 100644
index 0000000..850334e
--- /dev/null
+++ b/migrations/Version20230524205355.php
@@ -0,0 +1,71 @@
+addSql('CREATE TABLE educations (id SERIAL NOT NULL, institution VARCHAR(175) NOT NULL, faculty VARCHAR(175) NOT NULL, specialization VARCHAR(175) NOT NULL, from_date TIMESTAMP(0) WITHOUT TIME ZONE NOT NULL, to_date TIMESTAMP(0) WITHOUT TIME ZONE DEFAULT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE TABLE files (id SERIAL NOT NULL, name VARCHAR(200) NOT NULL, path VARCHAR(200) NOT NULL, mime_type VARCHAR(25) NOT NULL, size INT DEFAULT 0 NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE TABLE jobs (id SERIAL NOT NULL, name VARCHAR(175) NOT NULL, type VARCHAR(30) NOT NULL, description TEXT NOT NULL, from_date TIMESTAMP(0) WITHOUT TIME ZONE NOT NULL, to_date TIMESTAMP(0) WITHOUT TIME ZONE DEFAULT NULL, company_name VARCHAR(255) NOT NULL, company_url VARCHAR(255) NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE TABLE links (id SERIAL NOT NULL, project_id INT NOT NULL, title VARCHAR(175) NOT NULL, url VARCHAR(175) NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE INDEX IDX_D182A118166D1F9C ON links (project_id)');
+ $this->addSql('CREATE TABLE projects (id SERIAL NOT NULL, image_id INT DEFAULT NULL, name VARCHAR(175) NOT NULL, description TEXT NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE UNIQUE INDEX UNIQ_5C93B3A43DA5256D ON projects (image_id)');
+ $this->addSql('CREATE TABLE project_tags (project_id INT NOT NULL, tag_id INT NOT NULL, PRIMARY KEY(project_id, tag_id))');
+ $this->addSql('CREATE INDEX IDX_562D5C3E166D1F9C ON project_tags (project_id)');
+ $this->addSql('CREATE INDEX IDX_562D5C3EBAD26311 ON project_tags (tag_id)');
+ $this->addSql('CREATE TABLE properties (key VARCHAR(200) NOT NULL, value VARCHAR(200) NOT NULL, PRIMARY KEY(key))');
+ $this->addSql('CREATE TABLE sections (id SERIAL NOT NULL, name VARCHAR(175) NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE TABLE skills (id SERIAL NOT NULL, image_id INT DEFAULT NULL, section_id INT NOT NULL, name VARCHAR(175) NOT NULL, description TEXT NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE UNIQUE INDEX UNIQ_D53116703DA5256D ON skills (image_id)');
+ $this->addSql('CREATE INDEX IDX_D5311670D823E37A ON skills (section_id)');
+ $this->addSql('CREATE TABLE tags (id SERIAL NOT NULL, name VARCHAR(75) NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE TABLE users (id SERIAL NOT NULL, first_name VARCHAR(75) NOT NULL, last_name VARCHAR(75) NOT NULL, email VARCHAR(175) NOT NULL, password VARCHAR(175) NOT NULL, roles JSON NOT NULL, PRIMARY KEY(id))');
+ $this->addSql('CREATE UNIQUE INDEX UNIQ_1483A5E9E7927C74 ON users (email)');
+ $this->addSql('ALTER TABLE links ADD CONSTRAINT FK_D182A118166D1F9C FOREIGN KEY (project_id) REFERENCES projects (id) ON DELETE CASCADE NOT DEFERRABLE INITIALLY IMMEDIATE');
+ $this->addSql('ALTER TABLE projects ADD CONSTRAINT FK_5C93B3A43DA5256D FOREIGN KEY (image_id) REFERENCES files (id) NOT DEFERRABLE INITIALLY IMMEDIATE');
+ $this->addSql('ALTER TABLE project_tags ADD CONSTRAINT FK_562D5C3E166D1F9C FOREIGN KEY (project_id) REFERENCES projects (id) NOT DEFERRABLE INITIALLY IMMEDIATE');
+ $this->addSql('ALTER TABLE project_tags ADD CONSTRAINT FK_562D5C3EBAD26311 FOREIGN KEY (tag_id) REFERENCES tags (id) NOT DEFERRABLE INITIALLY IMMEDIATE');
+ $this->addSql('ALTER TABLE skills ADD CONSTRAINT FK_D53116703DA5256D FOREIGN KEY (image_id) REFERENCES files (id) NOT DEFERRABLE INITIALLY IMMEDIATE');
+ $this->addSql('ALTER TABLE skills ADD CONSTRAINT FK_D5311670D823E37A FOREIGN KEY (section_id) REFERENCES sections (id) ON DELETE CASCADE NOT DEFERRABLE INITIALLY IMMEDIATE');
+ }
+
+ public function down(Schema $schema): void
+ {
+ // this down() migration is auto-generated, please modify it to your needs
+ $this->addSql('CREATE SCHEMA public');
+ $this->addSql('ALTER TABLE links DROP CONSTRAINT FK_D182A118166D1F9C');
+ $this->addSql('ALTER TABLE projects DROP CONSTRAINT FK_5C93B3A43DA5256D');
+ $this->addSql('ALTER TABLE project_tags DROP CONSTRAINT FK_562D5C3E166D1F9C');
+ $this->addSql('ALTER TABLE project_tags DROP CONSTRAINT FK_562D5C3EBAD26311');
+ $this->addSql('ALTER TABLE skills DROP CONSTRAINT FK_D53116703DA5256D');
+ $this->addSql('ALTER TABLE skills DROP CONSTRAINT FK_D5311670D823E37A');
+ $this->addSql('DROP TABLE educations');
+ $this->addSql('DROP TABLE files');
+ $this->addSql('DROP TABLE jobs');
+ $this->addSql('DROP TABLE links');
+ $this->addSql('DROP TABLE projects');
+ $this->addSql('DROP TABLE project_tags');
+ $this->addSql('DROP TABLE properties');
+ $this->addSql('DROP TABLE sections');
+ $this->addSql('DROP TABLE skills');
+ $this->addSql('DROP TABLE tags');
+ $this->addSql('DROP TABLE users');
+ }
+}
diff --git a/phpcs.xml.dist b/phpcs.xml.dist
new file mode 100644
index 0000000..297d96a
--- /dev/null
+++ b/phpcs.xml.dist
@@ -0,0 +1,52 @@
+
+
+
+
+
+
+
+
+
+
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+ public/
+ src/
+ tests/
+
+
diff --git a/phpstan.neon.dist b/phpstan.neon.dist
new file mode 100644
index 0000000..ec605ef
--- /dev/null
+++ b/phpstan.neon.dist
@@ -0,0 +1,5 @@
+parameters:
+ level: 9
+ paths:
+ - src
+ - tests
diff --git a/phpunit.xml.dist b/phpunit.xml.dist
new file mode 100644
index 0000000..67777dd
--- /dev/null
+++ b/phpunit.xml.dist
@@ -0,0 +1,35 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ tests
+
+
+
+
+ src
+
+
+
+
+
diff --git a/public/index.php b/public/index.php
new file mode 100644
index 0000000..b85ed98
--- /dev/null
+++ b/public/index.php
@@ -0,0 +1,9 @@
+responseFactory->unauthorized("Bad credentials");
+ }
+
+ return $this->responseFactory->createResponse(new SystemMessage(200, "success login"));
+ }
+}
diff --git a/src/Controller/Auth/LogoutController.php b/src/Controller/Auth/LogoutController.php
new file mode 100644
index 0000000..b4b073c
--- /dev/null
+++ b/src/Controller/Auth/LogoutController.php
@@ -0,0 +1,29 @@
+security->logout(false);
+
+ return $this->responseFactory->createResponse();
+ }
+}
diff --git a/src/Controller/Education/CreateController.php b/src/Controller/Education/CreateController.php
new file mode 100644
index 0000000..b33fd18
--- /dev/null
+++ b/src/Controller/Education/CreateController.php
@@ -0,0 +1,43 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to create educations");
+ }
+
+ $result = $this->educationService->addEducation($educationData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(EducationView::create($result));
+ }
+}
diff --git a/src/Controller/Education/DeleteController.php b/src/Controller/Education/DeleteController.php
new file mode 100644
index 0000000..07f2add
--- /dev/null
+++ b/src/Controller/Education/DeleteController.php
@@ -0,0 +1,38 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to delete educations");
+ }
+
+ $education = $this->educationService->getEducationById($id);
+
+ if ($education === null) {
+ return $this->responseFactory->notFound("Education not found");
+ }
+
+ $this->educationService->deleteEducation($education);
+
+ return $this->responseFactory->createResponse();
+ }
+}
diff --git a/src/Controller/Education/ListController.php b/src/Controller/Education/ListController.php
new file mode 100644
index 0000000..bc91bfa
--- /dev/null
+++ b/src/Controller/Education/ListController.php
@@ -0,0 +1,43 @@
+educationService->getEducations($filter, $sort);
+ $count = $this->educationService->getCount($filter);
+
+ $collection = new CollectionChunk(
+ $filter->limit,
+ $filter->offset,
+ $count,
+ $educations
+ );
+
+ return $this->responseFactory->createResponse(
+ PaginatedView::create($collection, function (Education $education) {
+ return EducationView::create($education);
+ })
+ );
+ }
+}
diff --git a/src/Controller/Education/UpdateController.php b/src/Controller/Education/UpdateController.php
new file mode 100644
index 0000000..353f022
--- /dev/null
+++ b/src/Controller/Education/UpdateController.php
@@ -0,0 +1,46 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to modify educations");
+ }
+
+ $education = $this->educationService->getEducationById($id);
+
+ if ($education === null) {
+ return $this->responseFactory->notFound("Education not found");
+ }
+
+ $result = $this->educationService->updateEducation($education, $educationData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(EducationView::create($result));
+ }
+}
diff --git a/src/Controller/Education/ViewController.php b/src/Controller/Education/ViewController.php
new file mode 100644
index 0000000..523088f
--- /dev/null
+++ b/src/Controller/Education/ViewController.php
@@ -0,0 +1,30 @@
+educationService->getEducationById($id);
+
+ if ($education === null) {
+ return $this->responseFactory->notFound("Education not found");
+ }
+
+ return $this->responseFactory->createResponse(EducationView::create($education));
+ }
+}
diff --git a/src/Controller/Job/CreateController.php b/src/Controller/Job/CreateController.php
new file mode 100644
index 0000000..4e03fa2
--- /dev/null
+++ b/src/Controller/Job/CreateController.php
@@ -0,0 +1,43 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to create jobs");
+ }
+
+ $result = $this->jobService->addJob($jobData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(JobView::create($result));
+ }
+}
diff --git a/src/Controller/Job/DeleteController.php b/src/Controller/Job/DeleteController.php
new file mode 100644
index 0000000..67fafa5
--- /dev/null
+++ b/src/Controller/Job/DeleteController.php
@@ -0,0 +1,41 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to delete jobs");
+ }
+
+ $job = $this->jobService->getJobById($id);
+
+ if ($job === null) {
+ return $this->responseFactory->notFound("Job not found");
+ }
+
+ $this->jobService->deleteJob($job);
+
+ return $this->responseFactory->createResponse();
+ }
+}
diff --git a/src/Controller/Job/ListController.php b/src/Controller/Job/ListController.php
new file mode 100644
index 0000000..a0c78c4
--- /dev/null
+++ b/src/Controller/Job/ListController.php
@@ -0,0 +1,46 @@
+jobService->getJobs($filter, $sort);
+ $count = $this->jobService->getCount($filter);
+
+ $collection = new CollectionChunk(
+ $filter->limit,
+ $filter->offset,
+ $count,
+ $jobs
+ );
+
+ return $this->responseFactory->createResponse(
+ PaginatedView::create($collection, function (Job $job) {
+ return JobView::create($job);
+ })
+ );
+ }
+}
diff --git a/src/Controller/Job/UpdateController.php b/src/Controller/Job/UpdateController.php
new file mode 100644
index 0000000..a47f781
--- /dev/null
+++ b/src/Controller/Job/UpdateController.php
@@ -0,0 +1,49 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to modify jobs");
+ }
+
+ $job = $this->jobService->getJobById($id);
+
+ if ($job === null) {
+ return $this->responseFactory->notFound("Job not found");
+ }
+
+ $result = $this->jobService->updateJob($job, $jobData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(JobView::create($result));
+ }
+}
diff --git a/src/Controller/Job/ViewController.php b/src/Controller/Job/ViewController.php
new file mode 100644
index 0000000..7b856d1
--- /dev/null
+++ b/src/Controller/Job/ViewController.php
@@ -0,0 +1,33 @@
+jobService->getJobById($id);
+
+ if ($job === null) {
+ return $this->responseFactory->notFound("Job not found");
+ }
+
+ return $this->responseFactory->createResponse(JobView::create($job));
+ }
+}
diff --git a/src/Controller/Project/CreateController.php b/src/Controller/Project/CreateController.php
new file mode 100644
index 0000000..d5c9a5d
--- /dev/null
+++ b/src/Controller/Project/CreateController.php
@@ -0,0 +1,40 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to create projects");
+ }
+
+ $result = $this->projectService->addProject($projectData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(ProjectView::create($result));
+ }
+}
diff --git a/src/Controller/Project/DeleteController.php b/src/Controller/Project/DeleteController.php
new file mode 100644
index 0000000..991cd07
--- /dev/null
+++ b/src/Controller/Project/DeleteController.php
@@ -0,0 +1,38 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to delete projects");
+ }
+
+ $project = $this->projectService->getProjectById($id);
+
+ if ($project === null) {
+ return $this->responseFactory->notFound("Project not found");
+ }
+
+ $this->projectService->deleteProject($project);
+
+ return $this->responseFactory->createResponse();
+ }
+}
diff --git a/src/Controller/Project/ListController.php b/src/Controller/Project/ListController.php
new file mode 100644
index 0000000..ee60b3f
--- /dev/null
+++ b/src/Controller/Project/ListController.php
@@ -0,0 +1,40 @@
+projectService->getProjects($filter);
+ $count = $this->projectService->getCount($filter);
+
+ $collection = new CollectionChunk(
+ $filter->limit,
+ $filter->offset,
+ $count,
+ $projects,
+ );
+
+ return $this->responseFactory->createResponse(
+ PaginatedView::create($collection, fn (Project $project) => ProjectView::create($project))
+ );
+ }
+}
diff --git a/src/Controller/Project/UpdateController.php b/src/Controller/Project/UpdateController.php
new file mode 100644
index 0000000..369ce54
--- /dev/null
+++ b/src/Controller/Project/UpdateController.php
@@ -0,0 +1,47 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to modify projects");
+ }
+
+ $project = $this->projectService->getProjectById($id);
+
+ if ($project === null) {
+ return $this->responseFactory->notFound("Project not found");
+ }
+
+ $result = $this->projectService->updateProject($project, $projectData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(ProjectView::create($result));
+ }
+}
diff --git a/src/Controller/Project/ViewController.php b/src/Controller/Project/ViewController.php
new file mode 100644
index 0000000..34e0f67
--- /dev/null
+++ b/src/Controller/Project/ViewController.php
@@ -0,0 +1,30 @@
+projectService->getProjectById($id);
+
+ if ($project === null) {
+ return $this->responseFactory->notFound("Project not found");
+ }
+
+ return $this->responseFactory->createResponse(ProjectView::create($project));
+ }
+}
diff --git a/src/Controller/Property/ListController.php b/src/Controller/Property/ListController.php
new file mode 100644
index 0000000..5a5eab5
--- /dev/null
+++ b/src/Controller/Property/ListController.php
@@ -0,0 +1,27 @@
+propertyService->getProperties($propertyFilter);
+
+ return $this->responseFactory->createResponse(PropertiesView::create($properties));
+ }
+}
diff --git a/src/Controller/Property/UpdateController.php b/src/Controller/Property/UpdateController.php
new file mode 100644
index 0000000..1ac3989
--- /dev/null
+++ b/src/Controller/Property/UpdateController.php
@@ -0,0 +1,40 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to update configuration");
+ }
+
+ $result = $this->propertyService->updateProperties($propertiesData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(PropertiesView::create($result));
+ }
+}
diff --git a/src/Controller/Property/ViewController.php b/src/Controller/Property/ViewController.php
new file mode 100644
index 0000000..eefbda4
--- /dev/null
+++ b/src/Controller/Property/ViewController.php
@@ -0,0 +1,26 @@
+propertyService->getProperty($name);
+
+ return $this->responseFactory->createResponse(PropertiesView::create($configuration));
+ }
+}
diff --git a/src/Controller/Resume/GenerateController.php b/src/Controller/Resume/GenerateController.php
new file mode 100644
index 0000000..6ccf61c
--- /dev/null
+++ b/src/Controller/Resume/GenerateController.php
@@ -0,0 +1,21 @@
+projectService->execute();
+
+ }
+}
diff --git a/src/Controller/Skill/CreateController.php b/src/Controller/Skill/CreateController.php
new file mode 100644
index 0000000..d34ee3e
--- /dev/null
+++ b/src/Controller/Skill/CreateController.php
@@ -0,0 +1,40 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to create skills");
+ }
+
+ $result = $this->skillService->addSkill($skillData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(SkillView::create($result));
+ }
+}
diff --git a/src/Controller/Skill/DeleteController.php b/src/Controller/Skill/DeleteController.php
new file mode 100644
index 0000000..920dd3b
--- /dev/null
+++ b/src/Controller/Skill/DeleteController.php
@@ -0,0 +1,38 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to delete skills");
+ }
+
+ $skill = $this->skillService->getSkillById($id);
+
+ if ($skill === null) {
+ return $this->responseFactory->notFound("Skill not found");
+ }
+
+ $this->skillService->deleteSkill($skill);
+
+ return $this->responseFactory->createResponse();
+ }
+}
diff --git a/src/Controller/Skill/ListController.php b/src/Controller/Skill/ListController.php
new file mode 100644
index 0000000..28aa4ca
--- /dev/null
+++ b/src/Controller/Skill/ListController.php
@@ -0,0 +1,40 @@
+skillService->getSkills($filter);
+ $count = $this->skillService->getCount($filter);
+
+ $collection = new CollectionChunk(
+ $filter->limit,
+ $filter->offset,
+ $count,
+ $skills
+ );
+
+ return $this->responseFactory->createResponse(
+ PaginatedView::create($collection, fn (Skill $skill) => SkillView::create($skill))
+ );
+ }
+}
diff --git a/src/Controller/Skill/Section/CreateController.php b/src/Controller/Skill/Section/CreateController.php
new file mode 100644
index 0000000..9b20923
--- /dev/null
+++ b/src/Controller/Skill/Section/CreateController.php
@@ -0,0 +1,40 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to create skill sections");
+ }
+
+ $result = $this->sectionService->addSection($sectionData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(SectionView::create($result));
+ }
+}
diff --git a/src/Controller/Skill/Section/DeleteController.php b/src/Controller/Skill/Section/DeleteController.php
new file mode 100644
index 0000000..3557eb1
--- /dev/null
+++ b/src/Controller/Skill/Section/DeleteController.php
@@ -0,0 +1,38 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to delete sections");
+ }
+
+ $skill = $this->sectionService->getSectionById($id);
+
+ if ($skill === null) {
+ return $this->responseFactory->notFound("Section not found");
+ }
+
+ $this->sectionService->deleteSection($skill);
+
+ return $this->responseFactory->createResponse();
+ }
+}
diff --git a/src/Controller/Skill/Section/ListController.php b/src/Controller/Skill/Section/ListController.php
new file mode 100644
index 0000000..d7d5eb0
--- /dev/null
+++ b/src/Controller/Skill/Section/ListController.php
@@ -0,0 +1,40 @@
+sectionService->getSections($filter);
+ $count = $this->sectionService->getCount($filter);
+
+ $collection = new CollectionChunk(
+ $filter->limit,
+ $filter->offset,
+ $count,
+ $skills
+ );
+
+ return $this->responseFactory->createResponse(
+ PaginatedView::create($collection, fn (Section $section) => SectionView::create($section))
+ );
+ }
+}
diff --git a/src/Controller/Skill/Section/UpdateController.php b/src/Controller/Skill/Section/UpdateController.php
new file mode 100644
index 0000000..0b7d3af
--- /dev/null
+++ b/src/Controller/Skill/Section/UpdateController.php
@@ -0,0 +1,46 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to modify sections");
+ }
+
+ $section = $this->sectionService->getSectionById($id);
+
+ if ($section === null) {
+ return $this->responseFactory->notFound("Section not found");
+ }
+
+ $result = $this->sectionService->updateSection($sectionData, $section);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(SectionView::create($result));
+ }
+}
diff --git a/src/Controller/Skill/Section/ViewController.php b/src/Controller/Skill/Section/ViewController.php
new file mode 100644
index 0000000..185c04f
--- /dev/null
+++ b/src/Controller/Skill/Section/ViewController.php
@@ -0,0 +1,32 @@
+sectionService->getSectionById($id);
+
+ if ($section === null) {
+ return $this->responseFactory->notFound("Section not found");
+ }
+
+ return $this->responseFactory->createResponse(SectionView::create($section));
+ }
+}
diff --git a/src/Controller/Skill/UpdateController.php b/src/Controller/Skill/UpdateController.php
new file mode 100644
index 0000000..b0f015a
--- /dev/null
+++ b/src/Controller/Skill/UpdateController.php
@@ -0,0 +1,46 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to modify skills");
+ }
+
+ $skill = $this->skillService->getSkillById($id);
+
+ if ($skill === null) {
+ return $this->responseFactory->notFound("Skill not found");
+ }
+
+ $result = $this->skillService->updateSkill($skillData, $skill);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(SkillView::create($result));
+ }
+}
diff --git a/src/Controller/Skill/ViewController.php b/src/Controller/Skill/ViewController.php
new file mode 100644
index 0000000..3e3e90a
--- /dev/null
+++ b/src/Controller/Skill/ViewController.php
@@ -0,0 +1,30 @@
+skillService->getSkillById($id);
+
+ if ($skill === null) {
+ return $this->responseFactory->notFound("Skill not found");
+ }
+
+ return $this->responseFactory->createResponse(SkillView::create($skill));
+ }
+}
diff --git a/src/Controller/Storage/UploadController.php b/src/Controller/Storage/UploadController.php
new file mode 100644
index 0000000..bafe9a3
--- /dev/null
+++ b/src/Controller/Storage/UploadController.php
@@ -0,0 +1,50 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You can't upload files.");
+ }
+
+ /**
+ * @var UploadedFile|null $uploadedFile
+ */
+ $uploadedFile = $request->files->get("file");
+
+ if ($uploadedFile === null) {
+ return $this->responseFactory->badRequest("File not selected");
+ }
+
+ $result = $this->uploadService->execute($uploadedFile);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(FileView::create($result), Response::HTTP_CREATED);
+ }
+}
diff --git a/src/Controller/Storage/ViewController.php b/src/Controller/Storage/ViewController.php
new file mode 100644
index 0000000..66b8dea
--- /dev/null
+++ b/src/Controller/Storage/ViewController.php
@@ -0,0 +1,30 @@
+storageService->getFileById($id);
+
+ if ($file === null) {
+ return $this->responseFactory->notFound("File not found");
+ }
+
+ return $this->responseFactory->createResponse(FileView::create($file));
+ }
+}
diff --git a/src/Controller/User/DeleteController.php b/src/Controller/User/DeleteController.php
new file mode 100644
index 0000000..cae38a8
--- /dev/null
+++ b/src/Controller/User/DeleteController.php
@@ -0,0 +1,38 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to delete users");
+ }
+
+ $user = $this->userService->getUserById($id);
+
+ if ($user === null) {
+ return $this->responseFactory->notFound("User not found");
+ }
+
+ $this->userService->deleteUser($user);
+
+ return $this->responseFactory->createResponse();
+ }
+}
diff --git a/src/Controller/User/ListController.php b/src/Controller/User/ListController.php
new file mode 100644
index 0000000..97d961b
--- /dev/null
+++ b/src/Controller/User/ListController.php
@@ -0,0 +1,49 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You're not allowed to delete users");
+ }
+
+ $projects = $this->userService->getUsers($filter);
+ $count = $this->userService->getCount($filter);
+
+ $collection = new CollectionChunk(
+ $filter->limit,
+ $filter->offset,
+ $count,
+ $projects,
+ );
+
+ return $this->responseFactory->createResponse(
+ PaginatedView::create($collection, function (UserInterface $project) {
+ return UserView::create($project);
+ })
+ );
+ }
+}
diff --git a/src/Controller/User/RegistrationController.php b/src/Controller/User/RegistrationController.php
new file mode 100644
index 0000000..381776f
--- /dev/null
+++ b/src/Controller/User/RegistrationController.php
@@ -0,0 +1,36 @@
+registrationService->register($registrationData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(UserView::create($result), 201);
+ }
+}
diff --git a/src/Controller/User/UpdateController.php b/src/Controller/User/UpdateController.php
new file mode 100644
index 0000000..c4fffb1
--- /dev/null
+++ b/src/Controller/User/UpdateController.php
@@ -0,0 +1,46 @@
+security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You can't have access to modify user data");
+ }
+
+ $user = $this->userService->getUserById($id);
+
+ if ($user === null) {
+ return $this->responseFactory->notFound("User not found");
+ }
+
+ $result = $this->userService->updateUser($user, $userData);
+
+ if ($result instanceof ErrorInterface) {
+ return $this->responseFactory->createResponse(ErrorView::create($result), 400);
+ }
+
+ return $this->responseFactory->createResponse(UserView::create($result));
+ }
+}
diff --git a/src/Controller/User/ViewController.php b/src/Controller/User/ViewController.php
new file mode 100644
index 0000000..c598107
--- /dev/null
+++ b/src/Controller/User/ViewController.php
@@ -0,0 +1,43 @@
+responseFactory->createResponse(UserView::create($currentUser));
+ }
+
+ if (!$this->security->isGranted(Permissions::ADMIN)) {
+ return $this->responseFactory->forbidden("You cant have access to view user data");
+ }
+
+ $user = $this->userService->getUserById($id);
+
+ if ($user === null) {
+ return $this->responseFactory->notFound("User not found");
+ }
+
+ return $this->responseFactory->createResponse(UserView::create($currentUser));
+ }
+}
diff --git a/src/Entity/Company.php b/src/Entity/Company.php
new file mode 100644
index 0000000..dd179cf
--- /dev/null
+++ b/src/Entity/Company.php
@@ -0,0 +1,46 @@
+name;
+ }
+
+ /**
+ * @param string $name
+ */
+ public function setName(string $name): void
+ {
+ $this->name = $name;
+ }
+
+ /**
+ * @return string
+ */
+ public function getUrl(): string
+ {
+ return $this->url;
+ }
+
+ /**
+ * @param string $url
+ */
+ public function setUrl(string $url): void
+ {
+ $this->url = $url;
+ }
+}
diff --git a/src/Entity/Education.php b/src/Entity/Education.php
new file mode 100644
index 0000000..276b60e
--- /dev/null
+++ b/src/Entity/Education.php
@@ -0,0 +1,113 @@
+id = 0;
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getInstitution(): string
+ {
+ return $this->institution;
+ }
+
+ /**
+ * @param string $institution
+ */
+ public function changeInstitution(string $institution): void
+ {
+ $this->institution = $institution;
+ }
+
+ /**
+ * @return string
+ */
+ public function getFaculty(): string
+ {
+ return $this->faculty;
+ }
+
+ /**
+ * @param string $faculty
+ */
+ public function changeFaculty(string $faculty): void
+ {
+ $this->faculty = $faculty;
+ }
+
+ /**
+ * @return string
+ */
+ public function getSpecialization(): string
+ {
+ return $this->specialization;
+ }
+
+ /**
+ * @param string $specialization
+ */
+ public function changeSpecialization(string $specialization): void
+ {
+ $this->specialization = $specialization;
+ }
+
+ /**
+ * @return DateTimeInterface
+ */
+ public function getFromDate(): DateTimeInterface
+ {
+ return $this->fromDate;
+ }
+
+ /**
+ * @param DateTimeInterface $fromDate
+ */
+ public function changeFromDate(DateTimeInterface $fromDate): void
+ {
+ $this->fromDate = $fromDate;
+ }
+
+ /**
+ * @return ?DateTimeInterface
+ */
+ public function getToDate(): ?DateTimeInterface
+ {
+ return $this->toDate;
+ }
+
+ /**
+ * @param ?DateTimeInterface $toDate
+ */
+ public function changeToDate(?DateTimeInterface $toDate): void
+ {
+ $this->toDate = $toDate;
+ }
+}
diff --git a/src/Entity/File.php b/src/Entity/File.php
new file mode 100644
index 0000000..ca7ca49
--- /dev/null
+++ b/src/Entity/File.php
@@ -0,0 +1,94 @@
+id = 0;
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getName(): string
+ {
+ return $this->name;
+ }
+
+ /**
+ * @param string $name
+ */
+ public function setName(string $name): void
+ {
+ $this->name = $name;
+ }
+
+ /**
+ * @return string
+ */
+ public function getPath(): string
+ {
+ return $this->path;
+ }
+
+ /**
+ * @param string $path
+ */
+ public function setPath(string $path): void
+ {
+ $this->path = $path;
+ }
+
+ /**
+ * @return string
+ */
+ public function getMimeType(): string
+ {
+ return $this->mimeType;
+ }
+
+ /**
+ * @param string $mimeType
+ */
+ public function setMimeType(string $mimeType): void
+ {
+ $this->mimeType = $mimeType;
+ }
+
+ /**
+ * @return int
+ */
+ public function getSize(): int
+ {
+ return $this->size;
+ }
+
+ /**
+ * @param int $size
+ */
+ public function setSize(int $size): void
+ {
+ $this->size = $size;
+ }
+}
diff --git a/src/Entity/Job.php b/src/Entity/Job.php
new file mode 100644
index 0000000..155e0cb
--- /dev/null
+++ b/src/Entity/Job.php
@@ -0,0 +1,130 @@
+id = 0;
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getName(): string
+ {
+ return $this->name;
+ }
+
+ /**
+ * @param string $name
+ */
+ public function changeName(string $name): void
+ {
+ $this->name = $name;
+ }
+
+ /**
+ * @return string
+ */
+ public function getType(): string
+ {
+ return $this->type;
+ }
+
+ /**
+ * @param string $type
+ */
+ public function changeType(string $type): void
+ {
+ $this->type = $type;
+ }
+
+ /**
+ * @return string
+ */
+ public function getDescription(): string
+ {
+ return $this->description;
+ }
+
+ /**
+ * @param string $description
+ */
+ public function changeDescription(string $description): void
+ {
+ $this->description = $description;
+ }
+
+ /**
+ * @return Company
+ */
+ public function getCompany(): Company
+ {
+ return $this->company;
+ }
+
+ /**
+ * @param Company $company
+ */
+ public function changeCompany(Company $company): void
+ {
+ $this->company = $company;
+ }
+
+ /**
+ * @return DateTimeInterface
+ */
+ public function getFromDate(): DateTimeInterface
+ {
+ return $this->fromDate;
+ }
+
+ /**
+ * @param DateTimeInterface $fromDate
+ */
+ public function changeFromDate(DateTimeInterface $fromDate): void
+ {
+ $this->fromDate = $fromDate;
+ }
+
+ /**
+ * @return ?DateTimeInterface
+ */
+ public function getToDate(): ?DateTimeInterface
+ {
+ return $this->toDate;
+ }
+
+ /**
+ * @param ?DateTimeInterface $toDate
+ */
+ public function changeToDate(?DateTimeInterface $toDate): void
+ {
+ $this->toDate = $toDate;
+ }
+}
diff --git a/src/Entity/Link.php b/src/Entity/Link.php
new file mode 100644
index 0000000..7108001
--- /dev/null
+++ b/src/Entity/Link.php
@@ -0,0 +1,69 @@
+id = 0;
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return Project
+ */
+ public function getProject(): Project
+ {
+ return $this->project;
+ }
+
+ /**
+ * @return string
+ */
+ public function getTitle(): string
+ {
+ return $this->title;
+ }
+
+ /**
+ * @param string $title
+ */
+ public function changeTitle(string $title): void
+ {
+ $this->title = $title;
+ }
+
+ /**
+ * @return string
+ */
+ public function getUrl(): string
+ {
+ return $this->url;
+ }
+
+ /**
+ * @param string $url
+ */
+ public function changeUrl(string $url): void
+ {
+ $this->url = $url;
+ }
+}
diff --git a/src/Entity/Project.php b/src/Entity/Project.php
new file mode 100644
index 0000000..f0f279e
--- /dev/null
+++ b/src/Entity/Project.php
@@ -0,0 +1,177 @@
+
+ */
+ private Collection $links;
+
+ /**
+ * @var Collection
+ */
+ private Collection $tags;
+
+ /**
+ * @param string $name
+ * @param string $description
+ */
+ public function __construct(
+ private string $name,
+ private string $description,
+ private File $image,
+ ) {
+ $this->id = 0;
+ $this->links = new ArrayCollection();
+ $this->tags = new ArrayCollection();
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getName(): string
+ {
+ return $this->name;
+ }
+
+ /**
+ * @param string $name
+ */
+ public function changeName(string $name): void
+ {
+ $this->name = $name;
+ }
+
+ /**
+ * @return File
+ */
+ public function getImage(): File
+ {
+ return $this->image;
+ }
+
+ /**
+ * @param File $image
+ */
+ public function changeImage(File $image): void
+ {
+ $this->image = $image;
+ }
+
+ /**
+ * @return string
+ */
+ public function getDescription(): string
+ {
+ return $this->description;
+ }
+
+ /**
+ * @param string $description
+ */
+ public function changeDescription(string $description): void
+ {
+ $this->description = $description;
+ }
+
+ /**
+ * Add link
+ *
+ * @param Link $link
+ *
+ * @return void
+ */
+ public function addLink(Link $link): void
+ {
+ if (!$this->links->contains($link)) {
+ $this->links->add($link);
+ }
+ }
+
+ /**
+ * Remove link
+ *
+ * @param Link $link
+ *
+ * @return void
+ */
+ public function deleteLink(Link $link): void
+ {
+ if ($this->links->contains($link)) {
+ $this->links->removeElement($link);
+ }
+ }
+
+ /**
+ * Set links
+ *
+ * @param iterable $links
+ *
+ * @return void
+ */
+ public function setLinks(iterable $links): void
+ {
+ $this->links->clear();
+
+ foreach ($links as $link) {
+ $this->links->add($link);
+ }
+ }
+
+ /**
+ * @return Link[]
+ */
+ public function getLinks(): array
+ {
+ return $this->links->toArray();
+ }
+
+ public function addTag(Tag $tag): void
+ {
+ if (!$this->tags->contains($tag)) {
+ $this->tags->add($tag);
+ }
+ }
+
+ /**
+ * @param iterable $tags
+ *
+ * @return void
+ */
+ public function setTags(iterable $tags): void
+ {
+ $this->tags->clear();
+
+ foreach ($tags as $tag) {
+ $this->tags->add($tag);
+ }
+ }
+
+ /**
+ * @return Tag[]
+ */
+ public function getTags(): array
+ {
+ return $this->tags->toArray();
+ }
+}
diff --git a/src/Entity/Property.php b/src/Entity/Property.php
new file mode 100644
index 0000000..5f9bf51
--- /dev/null
+++ b/src/Entity/Property.php
@@ -0,0 +1,46 @@
+key;
+ }
+
+ /**
+ * @param string $key
+ */
+ public function setName(string $key): void
+ {
+ $this->key = $key;
+ }
+
+ /**
+ * @return string
+ */
+ public function getValue(): string
+ {
+ return $this->value;
+ }
+
+ /**
+ * @param string $value
+ */
+ public function setValue(string $value): void
+ {
+ $this->value = $value;
+ }
+}
diff --git a/src/Entity/Section.php b/src/Entity/Section.php
new file mode 100644
index 0000000..1472cde
--- /dev/null
+++ b/src/Entity/Section.php
@@ -0,0 +1,63 @@
+
+ */
+ private Collection $skills;
+
+ public function __construct(
+ private string $name,
+ ) {
+ $this->id = 0;
+ $this->skills = new ArrayCollection();
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getName(): string
+ {
+ return $this->name;
+ }
+
+ /**
+ * @param string $name
+ */
+ public function changeName(string $name): void
+ {
+ $this->name = $name;
+ }
+
+ /**
+ * @return Collection
+ */
+ public function getSkills(): Collection
+ {
+ return $this->skills;
+ }
+}
diff --git a/src/Entity/Skill.php b/src/Entity/Skill.php
new file mode 100644
index 0000000..89a5374
--- /dev/null
+++ b/src/Entity/Skill.php
@@ -0,0 +1,102 @@
+id = 0;
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getName(): string
+ {
+ return $this->name;
+ }
+
+ /**
+ * @param string $name
+ */
+ public function changeName(string $name): void
+ {
+ $this->name = $name;
+ }
+
+ /**
+ * @return Section
+ */
+ public function getSection(): Section
+ {
+ return $this->section;
+ }
+
+ /**
+ * @param Section $section
+ */
+ public function changeSection(Section $section): void
+ {
+ $this->section = $section;
+ }
+
+ /**
+ * @return string
+ */
+ public function getDescription(): string
+ {
+ return $this->description;
+ }
+
+ /**
+ * @param string $description
+ */
+ public function changeDescription(string $description): void
+ {
+ $this->description = $description;
+ }
+
+ /**
+ * @return File
+ */
+ public function getImage(): File
+ {
+ return $this->image;
+ }
+
+ /**
+ * @param File $image
+ */
+ public function changeImage(File $image): void
+ {
+ $this->image = $image;
+ }
+}
diff --git a/src/Entity/Tag.php b/src/Entity/Tag.php
new file mode 100644
index 0000000..005b1e3
--- /dev/null
+++ b/src/Entity/Tag.php
@@ -0,0 +1,35 @@
+id = 0;
+ }
+
+ /**
+ * @return int
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getName(): string
+ {
+ return $this->name;
+ }
+}
diff --git a/src/Entity/User.php b/src/Entity/User.php
new file mode 100644
index 0000000..9ea4dcb
--- /dev/null
+++ b/src/Entity/User.php
@@ -0,0 +1,133 @@
+id = 0;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getId(): int
+ {
+ return $this->id;
+ }
+
+ /**
+ * @return string
+ */
+ public function getEmail(): string
+ {
+ return $this->email;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function changeEmail(string $email): void
+ {
+ $this->email = $email;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getPassword(): string
+ {
+ return $this->password;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function changePassword(string $password): void
+ {
+ $this->password = $password;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getFirstName(): string
+ {
+ return $this->firstName;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function changeFirstName(string $firstName): void
+ {
+ $this->firstName = $firstName;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getLastName(): string
+ {
+ return $this->lastName;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function changeLastName(string $lastName): void
+ {
+ $this->lastName = $lastName;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getRoles(): array
+ {
+ return $this->roles;
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function eraseCredentials(): void
+ {
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getUserIdentifier(): string
+ {
+ return $this->getEmail();
+ }
+}
diff --git a/src/EventSubscriber/ConvertHttpErrorToResponseSubscriber.php b/src/EventSubscriber/ConvertHttpErrorToResponseSubscriber.php
new file mode 100644
index 0000000..0ade350
--- /dev/null
+++ b/src/EventSubscriber/ConvertHttpErrorToResponseSubscriber.php
@@ -0,0 +1,65 @@
+ */
+ public static function getSubscribedEvents(): array
+ {
+ return [ExceptionEvent::class => 'onKernelError'];
+ }
+
+ public function onKernelError(ExceptionEvent $event): void
+ {
+ if (!$event->isMainRequest()) {
+ return;
+ }
+
+ $error = $event->getThrowable();
+ switch (true) {
+ case $error instanceof NotFoundHttpException:
+ $response = $this->responseFactory->notFound('Resource is not found');
+ break;
+ case $error instanceof BadRequestHttpException:
+ $response = $this->responseFactory->badRequest($error->getMessage());
+ break;
+ case $error instanceof AccessDeniedHttpException:
+ $response = $this->responseFactory->forbidden('Forbidden access');
+ break;
+ case $error instanceof UnauthorizedHttpException:
+ $response = $this->responseFactory->unauthorized('Authorization is required');
+ break;
+ case $error instanceof HttpException:
+ $response = $this->responseFactory->createResponse(
+ new SystemMessage(
+ $error->getStatusCode(),
+ $error->getMessage(),
+ ),
+ $error->getStatusCode(),
+ );
+ break;
+ default:
+ return;
+ }
+
+ $event->setResponse($response);
+ }
+}
diff --git a/src/Kernel.php b/src/Kernel.php
new file mode 100644
index 0000000..ecc8737
--- /dev/null
+++ b/src/Kernel.php
@@ -0,0 +1,11 @@
+entityManager->persist($education);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function delete(Education $education): void
+ {
+ $this->entityManager->remove($education);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findOneById(int $id): ?Education
+ {
+ return $this->getRepository()->findOneBy(["id" => $id]);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getEducations(EducationFilter $filter, EducationSort $sort): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('education')
+ ->from(Education::class, 'education');
+
+ if ($filter->limit !== 0) {
+ $query->setMaxResults($filter->limit);
+ }
+
+ if ($filter->offset !== 0) {
+ $query->setFirstResult($filter->offset);
+ }
+
+ $field = $this->getSortField($sort->field);
+
+ $query->addOrderBy($field, $sort->order);
+
+ /** @var array $jobs */
+ $jobs = $query
+ ->getQuery()
+ ->getResult();
+
+ return new ArrayCollection($jobs);
+ }
+
+ /**
+ * @inheritDoc
+ *
+ * @throws NoResultException
+ * @throws NonUniqueResultException
+ */
+ public function getCount(EducationFilter $filter): int
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('COUNT(education)')
+ ->from(Education::class, 'education');
+
+ // @phpstan-ignore-next-line
+ return (int) $query->getQuery()
+ ->getSingleScalarResult();
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(Education::class);
+ }
+
+ private function createQueryBuilder(): QueryBuilder
+ {
+ return $this->entityManager->createQueryBuilder();
+ }
+
+ /**
+ * @param string $fieldName
+ *
+ * @return string
+ */
+ public function getSortField(string $fieldName): string
+ {
+ return match ($fieldName) {
+ 'id' => 'education.id',
+ 'name' => 'education.name',
+ 'from' => 'education.fromDate',
+ default => 'id',
+ };
+ }
+}
diff --git a/src/Repository/JobRepository.php b/src/Repository/JobRepository.php
new file mode 100644
index 0000000..9fe803d
--- /dev/null
+++ b/src/Repository/JobRepository.php
@@ -0,0 +1,130 @@
+entityManager->persist($job);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function delete(Job $job): void
+ {
+ $this->entityManager->remove($job);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findOneById(int $id): ?Job
+ {
+ return $this->getRepository()->findOneBy(["id" => $id]);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getJobs(JobFilter $filter, JobSort $sort): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('job')
+ ->from(Job::class, 'job');
+
+ if ($filter->limit !== null) {
+ $query->setMaxResults($filter->limit);
+ }
+
+ if ($filter->offset !== 0) {
+ $query->setFirstResult($filter->offset);
+ }
+
+ $field = $this->getSortField($sort->field);
+
+ $query->addOrderBy($field, $sort->order);
+
+ /** @var array $jobs */
+ $jobs = $query
+ ->getQuery()
+ ->getResult();
+
+ return new ArrayCollection($jobs);
+ }
+
+ /**
+ * @inheritDoc
+ *
+ * @throws NoResultException
+ * @throws NonUniqueResultException
+ */
+ public function getCount(JobFilter $filter): int
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('COUNT(j)')
+ ->from(Job::class, 'j');
+
+ // @phpstan-ignore-next-line
+ return (int) $query->getQuery()
+ ->getSingleScalarResult();
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(Job::class);
+ }
+
+ private function createQueryBuilder(): QueryBuilder
+ {
+ return $this->entityManager->createQueryBuilder();
+ }
+
+ /**
+ * @param string $fieldName
+ *
+ * @return string
+ */
+ public function getSortField(string $fieldName): string
+ {
+ return match ($fieldName) {
+ 'id' => 'job.id',
+ 'name' => 'job.name',
+ 'type' => 'job.type',
+ 'from' => 'job.fromDate',
+ 'company.name' => 'job.company.name',
+ default => 'id',
+ };
+ }
+}
diff --git a/src/Repository/LinkRepository.php b/src/Repository/LinkRepository.php
new file mode 100644
index 0000000..85ea596
--- /dev/null
+++ b/src/Repository/LinkRepository.php
@@ -0,0 +1,38 @@
+getRepository()->findBy(["project" => $project]);
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(Link::class);
+ }
+}
diff --git a/src/Repository/ProjectRepository.php b/src/Repository/ProjectRepository.php
new file mode 100644
index 0000000..91e35a0
--- /dev/null
+++ b/src/Repository/ProjectRepository.php
@@ -0,0 +1,147 @@
+entityManager->persist($project);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function delete(Project $project): void
+ {
+ $this->entityManager->remove($project);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ *
+ * @throws NonUniqueResultException
+ */
+ public function findOneById(int $id): ?Project
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('p', 't', 'l')
+ ->from(Project::class, 'p');
+
+ $query->leftJoin('p.tags', 't');
+ $query->leftJoin('p.links', 'l');
+ $query->where('p.id = :project');
+ $query->setParameter(':project', $id);
+
+ // @phpstan-ignore-next-line
+ return $query->getQuery()->getOneOrNullResult();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getProjects(ProjectFilter $filter): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('p')
+ ->from(Project::class, 'p');
+
+ $query->leftJoin('p.tags', 't')
+ ->addSelect('t');
+
+ $query->leftJoin('p.links', 'l')
+ ->addSelect('l');
+
+ $ids = $this->getProjectIds($filter);
+
+ /** @var array $projects */
+ $projects = $query
+ ->addOrderBy('p.id', 'ASC')
+ ->where('p.id IN(:projectIds)')
+ ->setParameter('projectIds', array_values($ids))
+ ->getQuery()
+ ->setHydrationMode(AbstractQuery::HYDRATE_ARRAY)
+ ->getResult();
+
+ return new ArrayCollection($projects);
+ }
+
+ /**
+ * @param ProjectFilter $filter
+ *
+ * @return int[]
+ */
+ private function getProjectIds(ProjectFilter $filter): array
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('DISTINCT p.id')
+ ->from(Project::class, 'p');
+
+ if ($filter->limit !== 0) {
+ $query->setMaxResults($filter->limit);
+ }
+
+ if ($filter->offset !== 0) {
+ $query->setFirstResult($filter->offset);
+ }
+
+ /** @var array $ids */
+ $ids = $query
+ ->addOrderBy('p.id', 'ASC')
+ ->getQuery()
+ ->getArrayResult();
+
+ return $ids;
+ }
+
+ /**
+ * @inheritDoc
+ *
+ * @throws NoResultException
+ * @throws NonUniqueResultException
+ */
+ public function getCount(ProjectFilter $filter): int
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('COUNT(p)')
+ ->from(Project::class, 'p');
+
+ // @phpstan-ignore-next-line
+ return (int) $query->getQuery()
+ ->getSingleScalarResult();
+ }
+
+ /**
+ * @return QueryBuilder
+ */
+ private function createQueryBuilder(): QueryBuilder
+ {
+ return $this->entityManager->createQueryBuilder();
+ }
+}
diff --git a/src/Repository/PropertyRepository.php b/src/Repository/PropertyRepository.php
new file mode 100644
index 0000000..9b5a857
--- /dev/null
+++ b/src/Repository/PropertyRepository.php
@@ -0,0 +1,116 @@
+getRepository()->findOneBy(["key" => $key]);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findByKeys(array $keys): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('prop')
+ ->from(Property::class, 'prop');
+
+ /** @var array $properties */
+ $properties = $query
+ ->where('prop.key IN(:keys)')
+ ->setParameter('keys', array_values($keys))
+ ->getQuery()
+ ->getResult();
+
+ return new ArrayCollection($properties);
+ }
+
+ public function getProperties(PropertyFilter $propertyFilter): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('prop')
+ ->from(Property::class, 'prop');
+
+ $fields = $propertyFilter->fields;
+
+ if (count($fields) > 0) {
+ $query->where('prop.key IN(:keys)')
+ ->setParameter('keys', $fields);
+ }
+
+
+ /** @var array $properties */
+ $properties = $query
+ ->getQuery()
+ ->getResult();
+
+ return new ArrayCollection($properties);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function save(Property $configuration): void
+ {
+ $this->entityManager->persist($configuration);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function saveMany(Collection $configurations): void
+ {
+ $this->entityManager->beginTransaction();
+
+ try {
+ foreach ($configurations as $configuration) {
+ $this->save($configuration);
+ }
+
+ $this->entityManager->commit();
+ } catch (Exception $e) {
+ $this->entityManager->rollBack();
+ throw $e;
+ }
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(Property::class);
+ }
+
+ /**
+ * @return QueryBuilder
+ */
+ private function createQueryBuilder(): QueryBuilder
+ {
+ return $this->entityManager->createQueryBuilder();
+ }
+}
diff --git a/src/Repository/SectionRepository.php b/src/Repository/SectionRepository.php
new file mode 100644
index 0000000..aae8aff
--- /dev/null
+++ b/src/Repository/SectionRepository.php
@@ -0,0 +1,110 @@
+entityManager->persist($section);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function delete(Section $section): void
+ {
+ $this->entityManager->remove($section);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findOneById(int $id): ?Section
+ {
+ return $this->getRepository()->findOneBy(["id" => $id]);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getSections(SectionFilter $filter): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('section')
+ ->from(Section::class, 'section');
+
+ $query->leftJoin('section.skills', 'skills')
+ ->addSelect('skills');
+
+ if ($filter->limit !== null) {
+ $query->setMaxResults($filter->limit);
+ }
+
+ if ($filter->offset !== 0) {
+ $query->setFirstResult($filter->offset);
+ }
+
+ $query->addOrderBy('section.id', 'ASC');
+
+ /** @var array $skills */
+ $skills = $query
+ ->getQuery()
+ ->getResult();
+
+ return new ArrayCollection($skills);
+ }
+
+ /**
+ * @inheritDoc
+ *
+ * @throws NoResultException
+ * @throws NonUniqueResultException
+ */
+ public function getCount(SectionFilter $filter): int
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('COUNT(s)')
+ ->from(Section::class, 's');
+
+ // @phpstan-ignore-next-line
+ return (int) $query->getQuery()
+ ->getSingleScalarResult();
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(Section::class);
+ }
+
+ private function createQueryBuilder(): QueryBuilder
+ {
+ return $this->entityManager->createQueryBuilder();
+ }
+}
diff --git a/src/Repository/SkillRepository.php b/src/Repository/SkillRepository.php
new file mode 100644
index 0000000..e90c700
--- /dev/null
+++ b/src/Repository/SkillRepository.php
@@ -0,0 +1,103 @@
+entityManager->persist($skill);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function delete(Skill $skill): void
+ {
+ $this->entityManager->remove($skill);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findOneById(int $id): ?Skill
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('skill', 'section', 'image')
+ ->from(Skill::class, 'skill');
+
+ $query->leftJoin('skill.section', 'section');
+ $query->leftJoin('skill.image', 'image');
+ $query->where('skill.id = :skill');
+ $query->setParameter(':skill', $id);
+
+ // @phpstan-ignore-next-line
+ return $query->getQuery()
+ ->getOneOrNullResult();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getSkills(SkillFilter $filter): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('s')
+ ->from(Skill::class, 's');
+
+ if ($filter->limit !== null) {
+ $query->setMaxResults($filter->limit);
+ }
+
+ if ($filter->offset !== 0) {
+ $query->setFirstResult($filter->offset);
+ }
+
+ /** @var array $skills */
+ $skills = $query
+ ->addOrderBy('s.id', 'ASC')
+ ->getQuery()
+ ->getResult();
+
+ return new ArrayCollection($skills);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getCount(SkillFilter $filter): int
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('COUNT(s)')
+ ->from(Skill::class, 's');
+
+ // @phpstan-ignore-next-line
+ return (int) $query->getQuery()
+ ->getSingleScalarResult();
+ }
+
+ private function createQueryBuilder(): QueryBuilder
+ {
+ return $this->entityManager->createQueryBuilder();
+ }
+}
diff --git a/src/Repository/StorageRepository.php b/src/Repository/StorageRepository.php
new file mode 100644
index 0000000..6e4b954
--- /dev/null
+++ b/src/Repository/StorageRepository.php
@@ -0,0 +1,45 @@
+entityManager->persist($file);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findOneById(int $id): ?File
+ {
+ return $this->getRepository()->findOneBy(['id' => $id]);
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(File::class);
+ }
+}
diff --git a/src/Repository/TagRepository.php b/src/Repository/TagRepository.php
new file mode 100644
index 0000000..57af974
--- /dev/null
+++ b/src/Repository/TagRepository.php
@@ -0,0 +1,51 @@
+getRepository()->findBy(["name" => $names]);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function saveTags(array $tags): array
+ {
+ foreach ($tags as $tag) {
+ $this->entityManager->persist($tag);
+ }
+
+ $this->entityManager->flush();
+
+ return $tags;
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(Tag::class);
+ }
+}
diff --git a/src/Repository/UserRepository.php b/src/Repository/UserRepository.php
new file mode 100644
index 0000000..2ca66a0
--- /dev/null
+++ b/src/Repository/UserRepository.php
@@ -0,0 +1,120 @@
+entityManager->persist($user);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function delete(UserInterface $user): void
+ {
+ $this->entityManager->remove($user);
+ $this->entityManager->flush();
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findOneByEmail(string $email): ?UserInterface
+ {
+ return $this->getRepository()->findOneBy(['email' => $email]);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function findOneById(int $id): ?UserInterface
+ {
+ return $this->getRepository()->findOneBy(['id' => $id]);
+ }
+
+ /**
+ * @inheritDoc
+ *
+ * @return Collection
+ */
+ public function getUsers(UserFilter $filter): Collection
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('u')
+ ->from(User::class, 'u');
+
+ if ($filter->limit !== null) {
+ $query->setMaxResults($filter->limit);
+ }
+
+ if ($filter->offset !== 0) {
+ $query->setFirstResult($filter->offset);
+ }
+
+ /** @var array $users */
+ $users = $query
+ ->addOrderBy('u.id', 'ASC')
+ ->getQuery()
+ ->getResult();
+
+ return new ArrayCollection($users);
+ }
+
+ /**
+ * @inheritDoc
+ *
+ * @throws NoResultException
+ * @throws NonUniqueResultException
+ */
+ public function getCount(UserFilter $filter): int
+ {
+ $query = $this->createQueryBuilder()
+ ->addSelect('COUNT(u)')
+ ->from(User::class, 'u');
+
+ // @phpstan-ignore-next-line
+ return (int) $query->getQuery()
+ ->getSingleScalarResult();
+ }
+
+ /**
+ * @return ObjectRepository
+ */
+ private function getRepository(): ObjectRepository
+ {
+ return $this->entityManager->getRepository(User::class);
+ }
+
+ private function createQueryBuilder(): QueryBuilder
+ {
+ return $this->entityManager->createQueryBuilder();
+ }
+}
diff --git a/src/Resources/Templates/Resume/resume.html.twig b/src/Resources/Templates/Resume/resume.html.twig
new file mode 100644
index 0000000..0dcd0ca
--- /dev/null
+++ b/src/Resources/Templates/Resume/resume.html.twig
@@ -0,0 +1,358 @@
+
+
+
+
+ {{ title }}
+
+
+
+
+
+
+
+ {% if about != null %}
+
+ {% endif %}
+
+ {% if sections|length > 0 %}
+
+
+
+ {% for key, section in sections %}
+
+
{{ section.name }}
+
+ {% for key, skill in section.skills %}
+ {{ skill.name }}{% if section.skills|length != key+1 %},{% endif %}
+ {% endfor %}
+
+
+ {% endfor %}
+
+
+ {% endif %}
+
+ {% if jobs|length > 0 %}
+
+
+
+ {% for key, job in jobs %}
+
+
+
+
{{ job.description }}
+
+
+ {% endfor %}
+
+
+ {% endif %}
+
+ {% if projects|length > 0 %}
+
+
+
+ {% for key, project in projects %}
+
+
+
+
{{ project.description }}
+
+
+ {% endfor %}
+
+
+ {% endif %}
+
+ {% if educations|length > 0 %}
+
+
+
+
+ {% for key, education in educations %}
+
+
+
+
{{ education.institution }}
+
+
+
+
{{ education.faculty }}
+
+
+
+ {{ education.fromDate|date('m-Y') }}/{% if education.toDate == null %}Present{% else %}{{ education.toDate|date('m-Y') }}{% endif %}
+
+
+
+
+
+ {{ education.specialization }}
+
+
+ {% endfor %}
+
+
+ {% endif %}
+
+
+
+
+
diff --git a/src/Security/AdminAccessVoter.php b/src/Security/AdminAccessVoter.php
new file mode 100644
index 0000000..8b56852
--- /dev/null
+++ b/src/Security/AdminAccessVoter.php
@@ -0,0 +1,44 @@
+getUser();
+
+ foreach ($attributes as $attribute) {
+ if ($attribute !== Permissions::ADMIN) {
+ continue;
+ }
+
+ if ($user === null) {
+ return self::ACCESS_DENIED;
+ }
+
+ if (!in_array("ROLE_ADMIN", $user->getRoles())) {
+ return self::ACCESS_DENIED;
+ }
+
+ $vote = self::ACCESS_GRANTED;
+ }
+
+ return $vote;
+ }
+}
diff --git a/src/Security/Permissions.php b/src/Security/Permissions.php
new file mode 100644
index 0000000..8c4b4df
--- /dev/null
+++ b/src/Security/Permissions.php
@@ -0,0 +1,10 @@
+month = $month;
+ $this->year = $year;
+ }
+}
diff --git a/src/Service/Education/Dto/EducationData.php b/src/Service/Education/Dto/EducationData.php
new file mode 100644
index 0000000..79e6c77
--- /dev/null
+++ b/src/Service/Education/Dto/EducationData.php
@@ -0,0 +1,42 @@
+institution = $institution;
+ $this->faculty = $faculty;
+ $this->specialization = $specialization;
+ $this->from = $from;
+ $this->to = $to;
+ }
+}
diff --git a/src/Service/Education/Dto/EducationFilter.php b/src/Service/Education/Dto/EducationFilter.php
new file mode 100644
index 0000000..9493492
--- /dev/null
+++ b/src/Service/Education/Dto/EducationFilter.php
@@ -0,0 +1,21 @@
+limit = $limit;
+ $this->offset = $offset;
+ }
+}
diff --git a/src/Service/Education/Dto/EducationSort.php b/src/Service/Education/Dto/EducationSort.php
new file mode 100644
index 0000000..952b2fa
--- /dev/null
+++ b/src/Service/Education/Dto/EducationSort.php
@@ -0,0 +1,39 @@
+field = $field;
+ }
+
+ if (in_array(strtoupper($order), self::AVAILABLE_ORDER)) {
+ $this->order = $order;
+ }
+ }
+}
diff --git a/src/Service/Education/EducationService.php b/src/Service/Education/EducationService.php
new file mode 100644
index 0000000..cfe9f3c
--- /dev/null
+++ b/src/Service/Education/EducationService.php
@@ -0,0 +1,153 @@
+validationService->validate($educationData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $toDate = null;
+
+ if ($educationData->to !== null) {
+ $toDate = $this->getDateFromDateData($educationData->to);
+ }
+
+ $education = new Education(
+ $educationData->institution,
+ $educationData->faculty,
+ $educationData->specialization,
+ $this->getDateFromDateData($educationData->from),
+ $toDate,
+ );
+
+ $this->educationRepository->save($education);
+
+ return $education;
+ }
+
+ /**
+ * Update education
+ *
+ * @param Education $education
+ * @param EducationData $educationData
+ *
+ * @return Education|ErrorInterface
+ */
+ public function updateEducation(Education $education, EducationData $educationData): Education|ErrorInterface
+ {
+ $validationError = $this->validationService->validate($educationData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $fromDate = $this->getDateFromDateData($educationData->from);
+
+ $toDate = null;
+
+ if ($educationData->to !== null) {
+ $toDate = $this->getDateFromDateData($educationData->to);
+ }
+
+ $education->changeInstitution($educationData->institution);
+ $education->changeFaculty($educationData->faculty);
+ $education->changeSpecialization($educationData->specialization);
+ $education->changeFromDate($fromDate);
+ $education->changeToDate($toDate);
+
+ $this->educationRepository->save($education);
+
+ return $education;
+ }
+
+ /**
+ * Get education by id
+ *
+ * @param int $id
+ *
+ * @return Education|null
+ */
+ public function getEducationById(int $id): ?Education
+ {
+ return $this->educationRepository->findOneById($id);
+ }
+
+ /**
+ * Delete education
+ *
+ * @param Education $education
+ *
+ * @return void
+ */
+ public function deleteEducation(Education $education): void
+ {
+ $this->educationRepository->delete($education);
+ }
+
+ /**
+ * Get list of educations
+ *
+ * @param EducationFilter $filter
+ * @param EducationSort $sort
+ *
+ * @return Collection
+ */
+ public function getEducations(EducationFilter $filter, EducationSort $sort): Collection
+ {
+ return $this->educationRepository->getEducations($filter, $sort);
+ }
+
+ /**
+ * Get count of educations
+ *
+ * @param EducationFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(EducationFilter $filter): int
+ {
+ return $this->educationRepository->getCount($filter);
+ }
+
+ /**
+ * @param DateData $dateData
+ *
+ * @return DateTime
+ */
+ private function getDateFromDateData(DateData $dateData): DateTime
+ {
+ return new DateTime("$dateData->year-$dateData->month");
+ }
+}
diff --git a/src/Service/Education/Repository/EducationRepositoryInterface.php b/src/Service/Education/Repository/EducationRepositoryInterface.php
new file mode 100644
index 0000000..333b550
--- /dev/null
+++ b/src/Service/Education/Repository/EducationRepositoryInterface.php
@@ -0,0 +1,59 @@
+
+ */
+ public function getEducations(EducationFilter $filter, EducationSort $sort): Collection;
+
+ /**
+ * Get educations count
+ *
+ * @param EducationFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(EducationFilter $filter): int;
+}
diff --git a/src/Service/Education/ValueResolver/InjectEducationFilter.php b/src/Service/Education/ValueResolver/InjectEducationFilter.php
new file mode 100644
index 0000000..54bc11d
--- /dev/null
+++ b/src/Service/Education/ValueResolver/InjectEducationFilter.php
@@ -0,0 +1,36 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if ($argumentType !== EducationFilter::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $limit = $request->query->getInt('limit', EducationFilter::EDUCATIONS_PER_PAGE);
+ $offset = $request->query->getInt('offset', 0);
+
+ yield new EducationFilter($limit, $offset);
+ }
+}
diff --git a/src/Service/Education/ValueResolver/InjectEducationSort.php b/src/Service/Education/ValueResolver/InjectEducationSort.php
new file mode 100644
index 0000000..d22a42d
--- /dev/null
+++ b/src/Service/Education/ValueResolver/InjectEducationSort.php
@@ -0,0 +1,48 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if ($argumentType !== EducationSort::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $sortJson = $request->query->get('sort') ?? '';
+
+ $field = EducationSort::DEFAULT_SORT;
+ $order = EducationSort::DEFAULT_ORDER;
+
+ if (!empty($sortJson)) {
+ /**
+ * @var object{field: string, order: string}|null $sort
+ */
+ $sort = json_decode((string)$sortJson, false);
+
+ $field = $sort?->field ?? EducationSort::DEFAULT_SORT;
+ $order = $sort?->order ?? $order;
+ }
+
+ yield new EducationSort($field, $order);
+ }
+}
diff --git a/src/Service/Error.php b/src/Service/Error.php
new file mode 100644
index 0000000..987b832
--- /dev/null
+++ b/src/Service/Error.php
@@ -0,0 +1,34 @@
+ $details
+ */
+ public function __construct(
+ private readonly string $message,
+ private readonly array $details = [],
+ ) {
+ }
+
+ /**
+ * @return string
+ */
+ public function getMessage(): string
+ {
+ return $this->message;
+ }
+
+ /**
+ * @return array
+ */
+ public function getDetails(): array
+ {
+ return $this->details;
+ }
+}
diff --git a/src/Service/ErrorInterface.php b/src/Service/ErrorInterface.php
new file mode 100644
index 0000000..b5f6577
--- /dev/null
+++ b/src/Service/ErrorInterface.php
@@ -0,0 +1,18 @@
+
+ */
+ public function getDetails(): array;
+}
diff --git a/src/Service/Job/Dto/CompanyData.php b/src/Service/Job/Dto/CompanyData.php
new file mode 100644
index 0000000..f5d101d
--- /dev/null
+++ b/src/Service/Job/Dto/CompanyData.php
@@ -0,0 +1,31 @@
+name = $name;
+ $this->url = $url;
+ }
+}
diff --git a/src/Service/Job/Dto/DateData.php b/src/Service/Job/Dto/DateData.php
new file mode 100644
index 0000000..90aca27
--- /dev/null
+++ b/src/Service/Job/Dto/DateData.php
@@ -0,0 +1,25 @@
+month = $month;
+ $this->year = $year;
+ }
+}
diff --git a/src/Service/Job/Dto/JobData.php b/src/Service/Job/Dto/JobData.php
new file mode 100644
index 0000000..7652cae
--- /dev/null
+++ b/src/Service/Job/Dto/JobData.php
@@ -0,0 +1,45 @@
+name = $name;
+ $this->description = $description;
+ $this->type = $type;
+ $this->company = $company;
+ $this->from = $from;
+ $this->to = $to;
+ }
+}
diff --git a/src/Service/Job/Dto/JobFilter.php b/src/Service/Job/Dto/JobFilter.php
new file mode 100644
index 0000000..517b3df
--- /dev/null
+++ b/src/Service/Job/Dto/JobFilter.php
@@ -0,0 +1,21 @@
+limit = $limit;
+ $this->offset = $offset;
+ }
+}
diff --git a/src/Service/Job/Dto/JobSort.php b/src/Service/Job/Dto/JobSort.php
new file mode 100644
index 0000000..422b65c
--- /dev/null
+++ b/src/Service/Job/Dto/JobSort.php
@@ -0,0 +1,40 @@
+field = $field;
+ }
+
+ if (in_array(strtoupper($order), self::AVAILABLE_ORDER)) {
+ $this->order = $order;
+ }
+ }
+}
diff --git a/src/Service/Job/JobService.php b/src/Service/Job/JobService.php
new file mode 100644
index 0000000..a23a059
--- /dev/null
+++ b/src/Service/Job/JobService.php
@@ -0,0 +1,159 @@
+validationService->validate($jobData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $company = $jobData->company;
+
+ $toDate = null;
+
+ if ($jobData->to !== null) {
+ $toDate = $this->getDateFromDateData($jobData->to);
+ }
+
+ $job = new Job(
+ $jobData->name,
+ $jobData->type,
+ $jobData->description,
+ new Company($company->name, $company->url),
+ $this->getDateFromDateData($jobData->from),
+ $toDate,
+ );
+
+ $this->jobRepository->save($job);
+
+ return $job;
+ }
+
+ /**
+ * Update job
+ *
+ * @param Job $job
+ * @param JobData $jobData
+ *
+ * @return Job|ErrorInterface
+ */
+ public function updateJob(Job $job, JobData $jobData): Job|ErrorInterface
+ {
+ $validationError = $this->validationService->validate($jobData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $toDate = null;
+
+ if ($jobData->to !== null) {
+ $toDate = $this->getDateFromDateData($jobData->to);
+ }
+
+ $company = $jobData->company;
+
+ $job->changeName($jobData->name);
+ $job->changeType($jobData->type);
+ $job->changeDescription($jobData->description);
+ $job->changeCompany(new Company($company->name, $company->url));
+ $job->changeFromDate($this->getDateFromDateData($jobData->from));
+ $job->changeToDate($toDate);
+
+ $this->jobRepository->save($job);
+
+ return $job;
+ }
+
+ /**
+ * Get Job by id
+ *
+ * @param int $id
+ *
+ * @return Job|null
+ */
+ public function getJobById(int $id): ?Job
+ {
+ return $this->jobRepository->findOneById($id);
+ }
+
+ /**
+ * Delete job
+ *
+ * @param Job $job
+ *
+ * @return void
+ */
+ public function deleteJob(Job $job): void
+ {
+ $this->jobRepository->delete($job);
+ }
+
+ /**
+ * Get list of jobs
+ *
+ * @param JobFilter $filter
+ * @param JobSort $sort
+ *
+ * @return Collection
+ */
+ public function getJobs(JobFilter $filter, JobSort $sort): Collection
+ {
+ return $this->jobRepository->getJobs($filter, $sort);
+ }
+
+ /**
+ * Get count of jobs
+ *
+ * @param JobFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(JobFilter $filter): int
+ {
+ return $this->jobRepository->getCount($filter);
+ }
+
+ /**
+ * @param DateData $dateData
+ *
+ * @return DateTime
+ */
+ private function getDateFromDateData(DateData $dateData): DateTime
+ {
+ return new DateTime("$dateData->year-$dateData->month");
+ }
+}
diff --git a/src/Service/Job/Repository/JobRepositoryInterface.php b/src/Service/Job/Repository/JobRepositoryInterface.php
new file mode 100644
index 0000000..a3adb2b
--- /dev/null
+++ b/src/Service/Job/Repository/JobRepositoryInterface.php
@@ -0,0 +1,59 @@
+
+ */
+ public function getJobs(JobFilter $filter, JobSort $sort): Collection;
+
+ /**
+ * Get jobs count
+ *
+ * @param JobFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(JobFilter $filter): int;
+}
diff --git a/src/Service/Job/ValueResolver/InjectJobFilter.php b/src/Service/Job/ValueResolver/InjectJobFilter.php
new file mode 100644
index 0000000..adeb3cd
--- /dev/null
+++ b/src/Service/Job/ValueResolver/InjectJobFilter.php
@@ -0,0 +1,36 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if ($argumentType !== JobFilter::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $limit = $request->query->getInt('limit', JobFilter::JOBS_PER_PAGE);
+ $offset = $request->query->getInt('offset', 0);
+
+ yield new JobFilter($limit, $offset);
+ }
+}
diff --git a/src/Service/Job/ValueResolver/InjectJobSort.php b/src/Service/Job/ValueResolver/InjectJobSort.php
new file mode 100644
index 0000000..4ee0e24
--- /dev/null
+++ b/src/Service/Job/ValueResolver/InjectJobSort.php
@@ -0,0 +1,48 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if ($argumentType !== JobSort::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $sortJson = $request->query->get('sort') ?? '';
+
+ $field = JobSort::DEFAULT_SORT;
+ $order = JobSort::DEFAULT_ORDER;
+
+ if (!empty($sortJson)) {
+ /**
+ * @var object{field: string, order: string}|null $sort
+ */
+ $sort = json_decode((string)$sortJson, false);
+
+ $field = $sort?->field ?? JobSort::DEFAULT_SORT;
+ $order = $sort?->order ?? $order;
+ }
+
+ yield new JobSort($field, $order);
+ }
+}
diff --git a/src/Service/Project/Dto/LinkData.php b/src/Service/Project/Dto/LinkData.php
new file mode 100644
index 0000000..5100e38
--- /dev/null
+++ b/src/Service/Project/Dto/LinkData.php
@@ -0,0 +1,22 @@
+limit = $limit;
+ $this->offset = $offset;
+ }
+}
diff --git a/src/Service/Project/Dto/ProjectSort.php b/src/Service/Project/Dto/ProjectSort.php
new file mode 100644
index 0000000..737e841
--- /dev/null
+++ b/src/Service/Project/Dto/ProjectSort.php
@@ -0,0 +1,34 @@
+field = $field;
+ }
+
+ if (in_array(strtoupper($order), self::AVAILABLE_ORDER)) {
+ $this->order = $order;
+ }
+ }
+}
diff --git a/src/Service/Project/LinkService.php b/src/Service/Project/LinkService.php
new file mode 100644
index 0000000..619bfc6
--- /dev/null
+++ b/src/Service/Project/LinkService.php
@@ -0,0 +1,56 @@
+
+ */
+ public function createLinks(Project $project, array $linksData): iterable
+ {
+ $existsLinks = [];
+ $newLinks = [];
+ $changedLink = [];
+ $links = $this->linkRepository->findLinksByProject($project);
+
+ foreach ($links as $link) {
+ $existsLinks[$link->getId()] = $link;
+ }
+
+ foreach ($linksData as $link) {
+ $id = $link->id;
+
+ if ($id !== null && isset($existsLinks[$id])) {
+ $item = $existsLinks[$id];
+
+ $item->changeTitle($link->title);
+ $item->changeUrl($link->url);
+
+ $changedLink[] = $item;
+ } else {
+ $newLinks[] = new Link($project, $link->title, $link->url);
+ }
+ }
+
+ yield from $changedLink;
+ yield from $newLinks;
+ }
+}
diff --git a/src/Service/Project/ProjectService.php b/src/Service/Project/ProjectService.php
new file mode 100644
index 0000000..8a37e9b
--- /dev/null
+++ b/src/Service/Project/ProjectService.php
@@ -0,0 +1,135 @@
+validationService->validate($projectData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $file = $this->storageService->getFileById($projectData->imageId);
+
+ if ($file === null) {
+ return new Error("Image id not found");
+ }
+
+ $tags = $this->tagService->createTags($projectData->tags);
+
+ $project = new Project(
+ $projectData->name,
+ $projectData->description,
+ $file
+ );
+
+ foreach ($tags as $tag) {
+ $project->addTag($tag);
+ }
+
+ foreach ($projectData->links as $link) {
+ $project->addLink(new Link($project, $link->title, $link->url));
+ }
+
+ $this->projectRepository->save($project);
+
+ return $project;
+ }
+
+ public function updateProject(Project $project, ProjectData $projectData): Project|ErrorInterface
+ {
+ $validationError = $this->validationService->validate($projectData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $file = $this->storageService->getFileById($projectData->imageId);
+
+ if ($file === null) {
+ return new Error("Image id not found");
+ }
+
+ $tags = $this->tagService->createTags($projectData->tags);
+ $links = $this->linkService->createLinks($project, $projectData->links);
+
+ $project->changeImage($file);
+ $project->changeName($projectData->name);
+ $project->changeDescription($projectData->description);
+ $project->setTags($tags);
+ $project->setLinks($links);
+
+ $this->projectRepository->save($project);
+
+ return $project;
+ }
+
+ /**
+ * @param ProjectFilter $filter
+ *
+ * @return Collection
+ */
+ public function getProjects(ProjectFilter $filter): Collection
+ {
+ return $this->projectRepository->getProjects($filter);
+ }
+
+ /**
+ * @param ProjectFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(ProjectFilter $filter): int
+ {
+ return $this->projectRepository->getCount($filter);
+ }
+
+ /**
+ * Delete project
+ *
+ * @param Project $project
+ *
+ * @return void
+ */
+ public function deleteProject(Project $project): void
+ {
+ $this->projectRepository->delete($project);
+ }
+
+ /**
+ * Get project by id
+ *
+ * @param int $id
+ *
+ * @return Project|null
+ */
+ public function getProjectById(int $id): ?Project
+ {
+ return $this->projectRepository->findOneById($id);
+ }
+}
diff --git a/src/Service/Project/Repository/LinkRepositoryInterface.php b/src/Service/Project/Repository/LinkRepositoryInterface.php
new file mode 100644
index 0000000..aca3ebb
--- /dev/null
+++ b/src/Service/Project/Repository/LinkRepositoryInterface.php
@@ -0,0 +1,20 @@
+
+ */
+ public function getProjects(ProjectFilter $filter): Collection;
+
+ /**
+ * Get count of projects
+ *
+ * @param ProjectFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(ProjectFilter $filter): int;
+}
diff --git a/src/Service/Project/Repository/TagRepositoryInterface.php b/src/Service/Project/Repository/TagRepositoryInterface.php
new file mode 100644
index 0000000..e8fe59d
--- /dev/null
+++ b/src/Service/Project/Repository/TagRepositoryInterface.php
@@ -0,0 +1,28 @@
+
+ */
+ public function createTags(array $names): iterable
+ {
+ $existingTags = $this->tagRepository->findTagsByNames($names);
+
+ $existingTagsNames = [];
+ foreach ($existingTags as $tag) {
+ $existingTagsNames[] = $tag->getName();
+ }
+
+ $newTags = [];
+ foreach ($names as $name) {
+ if (!in_array($name, $existingTagsNames, true)) {
+ $newTags[] = new Tag($name);
+ }
+ }
+
+ if (count($newTags) > 0) {
+ $this->tagRepository->saveTags($newTags);
+ }
+
+ yield from $existingTags;
+ yield from $newTags;
+ }
+}
diff --git a/src/Service/Project/ValueResolver/InjectProjectFilter.php b/src/Service/Project/ValueResolver/InjectProjectFilter.php
new file mode 100644
index 0000000..5e5640b
--- /dev/null
+++ b/src/Service/Project/ValueResolver/InjectProjectFilter.php
@@ -0,0 +1,43 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if (!$argumentType) {
+ return [];
+ }
+
+ if ($argumentType !== ProjectFilter::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+// $filter = $request->query->filter('filter');
+
+ $limit = $request->query->getInt('limit', ProjectFilter::PROJECTS_PER_PAGE);
+ $offset = $request->query->getInt('offset', 0);
+
+ yield new ProjectFilter($limit, $offset);
+ }
+}
diff --git a/src/Service/Property/Dto/PropertiesData.php b/src/Service/Property/Dto/PropertiesData.php
new file mode 100644
index 0000000..8fc5606
--- /dev/null
+++ b/src/Service/Property/Dto/PropertiesData.php
@@ -0,0 +1,18 @@
+fields = $fields;
+ }
+}
diff --git a/src/Service/Property/PropertyService.php b/src/Service/Property/PropertyService.php
new file mode 100644
index 0000000..62f3388
--- /dev/null
+++ b/src/Service/Property/PropertyService.php
@@ -0,0 +1,74 @@
+configurationRepository->findByKey($name);
+ }
+
+
+ public function getProperties(PropertyFilter $propertyFilter)
+ {
+ return $this->configurationRepository->getProperties($propertyFilter);
+ }
+
+ /**
+ * @param PropertiesData $configurationData
+ *
+ * @return Collection|ErrorInterface
+ */
+ public function updateProperties(PropertiesData $configurationData): Collection|ErrorInterface
+ {
+ $validationError = $this->validationService->validate($configurationData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $keys = [];
+ $propertiesData = [];
+
+ foreach ($configurationData->properties as $property) {
+ $key = $property->key;
+
+ $keys[] = $key;
+ $propertiesData[$key] = $property;
+ }
+
+ $properties = $this->configurationRepository->findByKeys($keys);
+
+ foreach ($properties as $property) {
+ $key = $property->getKey();
+ $property->setValue($propertiesData[$key]->value);
+
+ unset($propertiesData[$key]);
+ }
+
+ foreach ($propertiesData as $property) {
+ $properties->add(new Property($property->key, $property->value));
+ }
+
+ $this->configurationRepository->saveMany($properties);
+
+ return $properties;
+ }
+}
diff --git a/src/Service/Property/Repository/PropertyRepositoryInterface.php b/src/Service/Property/Repository/PropertyRepositoryInterface.php
new file mode 100644
index 0000000..a22e50b
--- /dev/null
+++ b/src/Service/Property/Repository/PropertyRepositoryInterface.php
@@ -0,0 +1,55 @@
+
+ */
+ public function findByKeys(array $keys): Collection;
+
+ /**
+ * Save configuration
+ *
+ * @param Property $configuration
+ *
+ * @return void
+ */
+ public function save(Property $configuration): void;
+
+ /**
+ * Save configurations
+ *
+ * @param Collection $configurations
+ *
+ * @return void
+ */
+ public function saveMany(Collection $configurations): void;
+
+ /**
+ * @param PropertyFilter $propertyFilter
+ *
+ * @return Collection
+ */
+ public function getProperties(PropertyFilter $propertyFilter): Collection;
+}
diff --git a/src/Service/Property/ValueResolver/InjectPropertyFilter.php b/src/Service/Property/ValueResolver/InjectPropertyFilter.php
new file mode 100644
index 0000000..5b6ec7e
--- /dev/null
+++ b/src/Service/Property/ValueResolver/InjectPropertyFilter.php
@@ -0,0 +1,53 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if (!$argumentType) {
+ return [];
+ }
+
+ if ($argumentType !== PropertyFilter::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $fields = $request->query->get('filter', null);
+ $filter = null;
+ if ($fields !== null) {
+ /**
+ * @var object $filter
+ */
+ $filter = json_decode((string)$fields, false);
+ }
+
+ $propertyFilter = new PropertyFilter();
+
+ if ($filter !== null && property_exists($filter, 'fields')) {
+ $propertyFilter->setFields($filter->fields);
+ }
+
+ yield $propertyFilter;
+ }
+}
diff --git a/src/Service/Response/Dto/CollectionChunk.php b/src/Service/Response/Dto/CollectionChunk.php
new file mode 100644
index 0000000..0bd2ab6
--- /dev/null
+++ b/src/Service/Response/Dto/CollectionChunk.php
@@ -0,0 +1,36 @@
+
+ */
+ public iterable $chunk;
+
+ /**
+ * @param int $limit
+ * @param int $offset
+ * @param int $ofTotalAmount
+ * @param iterable $chunk
+ */
+ public function __construct(int $limit, int $offset, int $ofTotalAmount, iterable $chunk)
+ {
+ $this->limit = $limit;
+ $this->offset = $offset;
+ $this->ofTotalAmount = $ofTotalAmount;
+ $this->chunk = $chunk;
+ }
+}
diff --git a/src/Service/Response/Dto/SystemMessage.php b/src/Service/Response/Dto/SystemMessage.php
new file mode 100644
index 0000000..08da595
--- /dev/null
+++ b/src/Service/Response/Dto/SystemMessage.php
@@ -0,0 +1,18 @@
+createResponse(new SystemMessage($code, $message), $code);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function notFound(string $message): Response
+ {
+ $code = Response::HTTP_NOT_FOUND;
+
+ return $this->createResponse(new SystemMessage($code, $message), $code);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function badRequest(string $message): Response
+ {
+ $code = Response::HTTP_BAD_REQUEST;
+
+ return $this->createResponse(new SystemMessage($code, $message), $code);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function unauthorized(string $message): Response
+ {
+ $code = Response::HTTP_UNAUTHORIZED;
+
+ return $this->createResponse(new SystemMessage($code, $message), $code);
+ }
+}
diff --git a/src/Service/Response/ResponseFactoryInterface.php b/src/Service/Response/ResponseFactoryInterface.php
new file mode 100644
index 0000000..2e00b83
--- /dev/null
+++ b/src/Service/Response/ResponseFactoryInterface.php
@@ -0,0 +1,64 @@
+ $headers
+ *
+ * @return Response
+ */
+ public function createResponse(
+ mixed $content = null,
+ int $statusCode = Response::HTTP_OK,
+ array $headers = []
+ ): Response;
+
+ /**
+ * Creates forbidden access response
+ *
+ * @param string $message
+ *
+ * @return Response
+ */
+ public function forbidden(string $message): Response;
+
+ /**
+ * Creates not-found response
+ *
+ * @param string $message
+ *
+ * @return Response
+ */
+ public function notFound(string $message): Response;
+
+ /**
+ * Creates bad-request response
+ *
+ * @param string $message
+ *
+ * @return Response
+ */
+ public function badRequest(string $message): Response;
+
+ /**
+ * Creates response for non authorized requests
+ *
+ * @param string $message
+ *
+ * @return Response
+ */
+ public function unauthorized(string $message): Response;
+}
diff --git a/src/Service/Resume/GenerateService.php b/src/Service/Resume/GenerateService.php
new file mode 100644
index 0000000..26ee4a7
--- /dev/null
+++ b/src/Service/Resume/GenerateService.php
@@ -0,0 +1,145 @@
+set('defaultFont', 'Arial');
+ $pdfOptions->set('enable_remote', true);
+
+ $dompdf = new Dompdf($pdfOptions);
+ $icons = $this->getIcons();
+
+ $sections = $this->sectionService->getSections(new SectionFilter(100, 0));
+ $educations = $this->educationService->getEducations(
+ new EducationFilter(100, 0),
+ new EducationSort('from', 'asc'),
+ );
+ $jobs = $this->jobService->getJobs(
+ new JobFilter(100, 0),
+ new JobSort('from', 'asc'),
+ );
+ $projects = $this->projectService->getProjects(
+ new ProjectFilter(100, 0),
+ );
+
+ $properties = $this->propertyService->getProperties(new PropertyFilter([]));
+
+ $links = [];
+ $photo = null;
+ $about = null;
+ foreach ($properties as $property) {
+ $key = $property->getKey();
+ $value = $property->getValue();
+
+ if (in_array($key, [
+ 'githubLink',
+ 'linkedinLink',
+ 'siteUrl',
+ 'emailAddress',
+ 'phoneNumber',
+ ])) {
+ $links[$key] = $value;
+ }
+ elseif ($key === 'userPhoto') {
+ $file = $this->storageService->getFileById((int)$value);
+
+ if ($file !== null) {
+ $base64 = $this->getBase64Image(
+ $_SERVER["DOCUMENT_ROOT"] . $file->getPath() . $file->getName(),
+ $file->getMimeType(),
+ );
+ if ($base64 != null) {
+ $photo = $base64;
+ }
+ }
+ }
+ elseif ($key === 'about') {
+ $about = $value;
+ }
+ }
+
+
+ $html = $this->twig->render('Resume/resume.html.twig', [
+ 'title' => "Sergey Rashin",
+ 'icons' => $icons,
+ 'sections' => $sections,
+ 'educations' => $educations,
+ 'jobs' => $jobs,
+ 'projects' => $projects,
+ 'links' => $links,
+ 'userPhoto' => $photo,
+ 'about' => $about,
+ ]);
+
+ $dompdf->loadHtml($html);
+
+ $dompdf->setPaper('A4');
+
+ $dompdf->render();
+
+ $dompdf->stream("SergeyRashin.pdf", [
+ "Attachment" => false
+ ]);
+ }
+
+ /**
+ * Dompdf not supports svg images or font icons
+ *
+ * @return string[]
+ */
+ public function getIcons(): array
+ {
+ // @codingStandardsIgnoreStart
+ return [
+ 'linkedin' => base64_encode(''),
+ 'github' => base64_encode(''),
+ 'phone' => base64_encode(''),
+ 'email' => base64_encode(''),
+ 'site' => base64_encode(''),
+ ];
+ // @codingStandardsIgnoreEnd
+ }
+
+ public function getBase64Image(string $path, string $ext): string|null
+ {
+ $data = file_get_contents($path);
+ if (!$data)
+ return null;
+
+ return 'data:' . $ext . ';base64,' . base64_encode($data);
+ }
+}
diff --git a/src/Service/Skill/Dto/SectionData.php b/src/Service/Skill/Dto/SectionData.php
new file mode 100644
index 0000000..f2f088d
--- /dev/null
+++ b/src/Service/Skill/Dto/SectionData.php
@@ -0,0 +1,14 @@
+limit = $limit;
+ $this->offset = $offset;
+ }
+}
diff --git a/src/Service/Skill/Dto/SkillData.php b/src/Service/Skill/Dto/SkillData.php
new file mode 100644
index 0000000..9c996b1
--- /dev/null
+++ b/src/Service/Skill/Dto/SkillData.php
@@ -0,0 +1,26 @@
+limit = $limit;
+ $this->offset = $offset;
+ }
+}
diff --git a/src/Service/Skill/Repository/SectionRepositoryInterface.php b/src/Service/Skill/Repository/SectionRepositoryInterface.php
new file mode 100644
index 0000000..b310a0b
--- /dev/null
+++ b/src/Service/Skill/Repository/SectionRepositoryInterface.php
@@ -0,0 +1,57 @@
+
+ */
+ public function getSections(SectionFilter $filter): Collection;
+
+ /**
+ * Get count of sections
+ *
+ * @param SectionFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(SectionFilter $filter): int;
+}
diff --git a/src/Service/Skill/Repository/SkillRepositoryInterface.php b/src/Service/Skill/Repository/SkillRepositoryInterface.php
new file mode 100644
index 0000000..876505a
--- /dev/null
+++ b/src/Service/Skill/Repository/SkillRepositoryInterface.php
@@ -0,0 +1,57 @@
+
+ */
+ public function getSkills(SkillFilter $filter): Collection;
+
+ /**
+ * Get count of skills
+ *
+ * @param SkillFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(SkillFilter $filter): int;
+}
diff --git a/src/Service/Skill/SectionService.php b/src/Service/Skill/SectionService.php
new file mode 100644
index 0000000..a9926eb
--- /dev/null
+++ b/src/Service/Skill/SectionService.php
@@ -0,0 +1,83 @@
+validationService->validate($sectionData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $section = new Section(
+ $sectionData->name
+ );
+
+ $this->sectionRepository->save($section);
+
+ return $section;
+ }
+
+ public function updateSection(SectionData $sectionData, Section $section): Section|ErrorInterface
+ {
+ $validationError = $this->validationService->validate($sectionData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $section->changeName($sectionData->name);
+
+ $this->sectionRepository->save($section);
+
+ return $section;
+ }
+
+ public function deleteSection(Section $section): void
+ {
+ $this->sectionRepository->delete($section);
+ }
+
+ public function getSectionById(int $sectionId): ?Section
+ {
+ return $this->sectionRepository->findOneById($sectionId);
+ }
+
+ /**
+ * @param SectionFilter $filter
+ *
+ * @return Collection
+ */
+ public function getSections(SectionFilter $filter): Collection
+ {
+ return $this->sectionRepository->getSections($filter);
+ }
+
+ /**
+ * @param SectionFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(SectionFilter $filter): int
+ {
+ return $this->sectionRepository->getCount($filter);
+ }
+}
diff --git a/src/Service/Skill/SkillService.php b/src/Service/Skill/SkillService.php
new file mode 100644
index 0000000..22e29a5
--- /dev/null
+++ b/src/Service/Skill/SkillService.php
@@ -0,0 +1,151 @@
+validationService->validate($skillData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $file = $this->storageService->getFileById($skillData->imageId);
+
+ if ($file === null) {
+ return new Error("Image id not found");
+ }
+
+ $section = $this->sectionService->getSectionById($skillData->sectionId);
+
+ if ($section === null) {
+ return new Error("Section id not found");
+ }
+
+ $skill = new Skill(
+ $skillData->name,
+ $section,
+ $file,
+ $skillData->description,
+ );
+
+ $this->skillRepository->save($skill);
+
+ return $skill;
+ }
+
+ /**
+ * Update skill
+ *
+ * @param SkillData $skillData
+ * @param Skill $skill
+ *
+ * @return Skill|ErrorInterface
+ */
+ public function updateSkill(SkillData $skillData, Skill $skill): Skill|ErrorInterface
+ {
+ $validationError = $this->validationService->validate($skillData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $file = $this->storageService->getFileById($skillData->imageId);
+
+ if ($file === null) {
+ return new Error("Image id not found");
+ }
+
+ $section = $this->sectionService->getSectionById($skillData->sectionId);
+
+ if ($section === null) {
+ return new Error("Section id not found");
+ }
+
+ $skill->changeName($skillData->name);
+ $skill->changeImage($file);
+ $skill->changeSection($section);
+ $skill->changeDescription($skillData->description);
+
+ $this->skillRepository->save($skill);
+
+ return $skill;
+ }
+
+ /**
+ * Delete skill
+ *
+ * @param Skill $skill
+ *
+ * @return void
+ */
+ public function deleteSkill(Skill $skill): void
+ {
+ $this->skillRepository->delete($skill);
+ }
+
+ /**
+ * Get skill by id
+ *
+ * @param int $id
+ *
+ * @return Skill|null
+ */
+ public function getSkillById(int $id): ?Skill
+ {
+ return $this->skillRepository->findOneById($id);
+ }
+
+ /**
+ * Get skills
+ *
+ * @param SkillFilter $filter
+ *
+ * @return Collection
+ */
+ public function getSkills(SkillFilter $filter): Collection
+ {
+ return $this->skillRepository->getSkills($filter);
+ }
+
+ /**
+ * Get skills count
+ *
+ * @param SkillFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(SkillFilter $filter): int
+ {
+ return $this->skillRepository->getCount($filter);
+ }
+}
diff --git a/src/Service/Skill/ValueResolver/InjectSectionFilter.php b/src/Service/Skill/ValueResolver/InjectSectionFilter.php
new file mode 100644
index 0000000..4ea3f03
--- /dev/null
+++ b/src/Service/Skill/ValueResolver/InjectSectionFilter.php
@@ -0,0 +1,36 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if ($argumentType !== SectionFilter::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $limit = $request->query->getInt('limit', SectionFilter::SECTIONS_PER_PAGE);
+ $offset = $request->query->getInt('offset', 0);
+
+ yield new SectionFilter($limit, $offset);
+ }
+}
diff --git a/src/Service/Skill/ValueResolver/InjectSkillFilter.php b/src/Service/Skill/ValueResolver/InjectSkillFilter.php
new file mode 100644
index 0000000..9c544ab
--- /dev/null
+++ b/src/Service/Skill/ValueResolver/InjectSkillFilter.php
@@ -0,0 +1,36 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if ($argumentType !== SkillFilter::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $limit = $request->query->getInt('limit', SkillFilter::SKILLS_PER_PAGE);
+ $offset = $request->query->getInt('offset', 0);
+
+ yield new SkillFilter($limit, $offset);
+ }
+}
diff --git a/src/Service/Storage/Repository/StorageRepositoryInterface.php b/src/Service/Storage/Repository/StorageRepositoryInterface.php
new file mode 100644
index 0000000..f448ff3
--- /dev/null
+++ b/src/Service/Storage/Repository/StorageRepositoryInterface.php
@@ -0,0 +1,28 @@
+storageRepository->findOneById($id);
+ }
+}
diff --git a/src/Service/Storage/UploadService.php b/src/Service/Storage/UploadService.php
new file mode 100644
index 0000000..41fb80a
--- /dev/null
+++ b/src/Service/Storage/UploadService.php
@@ -0,0 +1,57 @@
+getClientOriginalName(), PATHINFO_FILENAME);
+ $safeFilename = $this->slugger->slug($originalFilename);
+ $newFilename = $safeFilename . '-' . uniqid() . '.' . $file->getClientOriginalExtension();
+
+ $fileDirectory = $this->storagePublicPath . date('d-m-Y') . DIRECTORY_SEPARATOR;
+ $filePath = $this->storagePublicUrl . date('d-m-Y') . DIRECTORY_SEPARATOR;
+ $mime = $file->getMimeType() ?? '';
+ $size = $file->getSize() ?: 0 ;
+
+ $file->move(
+ $fileDirectory,
+ $newFilename
+ );
+
+ $file = new File(
+ $newFilename,
+ $filePath,
+ $mime,
+ $size,
+ );
+
+ $this->storageRepository->save($file);
+
+ return $file;
+ }
+}
diff --git a/src/Service/User/Constraint/UniqueEmail.php b/src/Service/User/Constraint/UniqueEmail.php
new file mode 100644
index 0000000..98ac172
--- /dev/null
+++ b/src/Service/User/Constraint/UniqueEmail.php
@@ -0,0 +1,23 @@
+limit = $limit;
+ $this->offset = $offset;
+ }
+}
diff --git a/src/Service/User/Model/UserInterface.php b/src/Service/User/Model/UserInterface.php
new file mode 100644
index 0000000..fe6c2a7
--- /dev/null
+++ b/src/Service/User/Model/UserInterface.php
@@ -0,0 +1,85 @@
+validationService->validate($registrationData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $user = new User(
+ $registrationData->email,
+ $registrationData->password,
+ $registrationData->firstName,
+ $registrationData->lastName,
+ );
+
+ $hashedPassword = $this->userPasswordHasher->hashPassword(
+ $user,
+ $registrationData->password
+ );
+
+ $user->changePassword($hashedPassword);
+
+ $this->userRepository->save($user);
+
+ return $user;
+ }
+}
diff --git a/src/Service/User/Repository/UserRepositoryInterface.php b/src/Service/User/Repository/UserRepositoryInterface.php
new file mode 100644
index 0000000..c2e7a1e
--- /dev/null
+++ b/src/Service/User/Repository/UserRepositoryInterface.php
@@ -0,0 +1,69 @@
+
+ */
+ public function getUsers(UserFilter $filter): Collection;
+
+ /**
+ * Get count of users
+ *
+ * @param UserFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(UserFilter $filter): int;
+}
diff --git a/src/Service/User/UserService.php b/src/Service/User/UserService.php
new file mode 100644
index 0000000..ab771ff
--- /dev/null
+++ b/src/Service/User/UserService.php
@@ -0,0 +1,70 @@
+validationService->validate($userData);
+
+ if ($validationError !== null) {
+ return $validationError;
+ }
+
+ $user->changeFirstName($userData->firstName);
+ $user->changeLastName($userData->lastName);
+
+ $hashedPassword = $this->userPasswordHasher->hashPassword(
+ $user,
+ $userData->password
+ );
+
+ $user->changePassword($hashedPassword);
+
+ $this->userRepository->save($user);
+
+ return $user;
+ }
+
+ public function deleteUser(UserInterface $user): void
+ {
+ $this->userRepository->delete($user);
+ }
+
+ public function getUserById(int $id): ?UserInterface
+ {
+ return $this->userRepository->findOneById($id);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function getUsers(UserFilter $filter): Collection
+ {
+ return $this->userRepository->getUsers($filter);
+ }
+
+ public function getCount(UserFilter $filter): int
+ {
+ return $this->userRepository->getCount($filter);
+ }
+}
diff --git a/src/Service/User/UserServiceInterface.php b/src/Service/User/UserServiceInterface.php
new file mode 100644
index 0000000..bfd3b27
--- /dev/null
+++ b/src/Service/User/UserServiceInterface.php
@@ -0,0 +1,64 @@
+
+ */
+ public function getUsers(UserFilter $filter): Collection;
+
+ /**
+ * Get count of users
+ *
+ * @param UserFilter $filter
+ *
+ * @return int
+ */
+ public function getCount(UserFilter $filter): int;
+}
diff --git a/src/Service/User/Validator/UniqueEmailValidator.php b/src/Service/User/Validator/UniqueEmailValidator.php
new file mode 100644
index 0000000..8801f54
--- /dev/null
+++ b/src/Service/User/Validator/UniqueEmailValidator.php
@@ -0,0 +1,39 @@
+userRepository->findOneByEmail($value);
+
+ if ($user !== null) {
+ $this->context->addViolation(
+ // @phpstan-ignore-next-line
+ $constraint->message,
+ [
+ 'value' => $value,
+ ],
+ );
+ }
+ }
+}
diff --git a/src/Service/User/ValueResolver/InjectUserFilter.php b/src/Service/User/ValueResolver/InjectUserFilter.php
new file mode 100644
index 0000000..701c634
--- /dev/null
+++ b/src/Service/User/ValueResolver/InjectUserFilter.php
@@ -0,0 +1,36 @@
+
+ */
+ public function resolve(Request $request, ArgumentMetadata $argument): iterable
+ {
+ $argumentType = $argument->getType();
+
+ if ($argumentType !== UserFilter::class) {
+ return [];
+ }
+
+ if (!class_exists($argumentType)) {
+ return [];
+ }
+
+ $limit = $request->query->getInt('limit', UserFilter::USERS_PER_PAGE);
+ $offset = $request->query->getInt('offset', 0);
+
+ yield new UserFilter($limit, $offset);
+ }
+}
diff --git a/src/Service/Validation/ValidationError.php b/src/Service/Validation/ValidationError.php
new file mode 100644
index 0000000..8b8e914
--- /dev/null
+++ b/src/Service/Validation/ValidationError.php
@@ -0,0 +1,21 @@
+ $details
+ */
+ public function __construct(
+ string $message,
+ array $details,
+ ) {
+ parent::__construct($message, $details);
+ }
+}
diff --git a/src/Service/Validation/ValidationService.php b/src/Service/Validation/ValidationService.php
new file mode 100644
index 0000000..07d0062
--- /dev/null
+++ b/src/Service/Validation/ValidationService.php
@@ -0,0 +1,38 @@
+validator->validate($value, $constraints);
+
+ if (count($violations) === 0) {
+ return null;
+ }
+
+ $errors = [];
+ foreach ($violations as $violation) {
+ $errors[$violation->getPropertyPath()] = (string) $violation->getMessage();
+ }
+
+ return new ValidationError('Validation Error', $errors);
+ }
+}
diff --git a/src/Service/Validation/ValidationServiceInterface.php b/src/Service/Validation/ValidationServiceInterface.php
new file mode 100644
index 0000000..ed87ecf
--- /dev/null
+++ b/src/Service/Validation/ValidationServiceInterface.php
@@ -0,0 +1,23 @@
+
+ */
+ public static function create(Education $education): array
+ {
+ return [
+ 'id' => $education->getId(),
+ 'institution' => $education->getInstitution(),
+ 'faculty' => $education->getFaculty(),
+ 'specialization' => $education->getSpecialization(),
+ 'from' => DateView::create($education->getFromDate()),
+ 'to' => $education->getToDate() ? DateView::create($education->getToDate()) : null,
+ ];
+ }
+}
diff --git a/src/View/ErrorView.php b/src/View/ErrorView.php
new file mode 100644
index 0000000..0b2fc2d
--- /dev/null
+++ b/src/View/ErrorView.php
@@ -0,0 +1,23 @@
+
+ */
+ public static function create(ErrorInterface $error): array
+ {
+ return [
+ 'message' => $error->getMessage(),
+ 'details' => $error->getDetails(),
+ ];
+ }
+}
diff --git a/src/View/FileView.php b/src/View/FileView.php
new file mode 100644
index 0000000..dc806ff
--- /dev/null
+++ b/src/View/FileView.php
@@ -0,0 +1,27 @@
+
+ */
+ public static function create(File $file): array
+ {
+ return [
+ 'id' => $file->getId(),
+ 'name' => $file->getName(),
+ 'path' => $file->getPath(),
+ 'src' => $file->getPath() . $file->getName(),
+ 'mimeType' => $file->getMimeType(),
+ 'size' => $file->getSize(),
+ ];
+ }
+}
diff --git a/src/View/ImageView.php b/src/View/ImageView.php
new file mode 100644
index 0000000..9b92827
--- /dev/null
+++ b/src/View/ImageView.php
@@ -0,0 +1,27 @@
+
+ */
+ public static function create(?File $file): array
+ {
+ if ($file === null) {
+ return [];
+ }
+
+ return [
+ 'id' => $file->getId(),
+ 'src' => $file->getPath() . $file->getName(),
+ ];
+ }
+}
diff --git a/src/View/Job/CompanyView.php b/src/View/Job/CompanyView.php
new file mode 100644
index 0000000..1a9be6f
--- /dev/null
+++ b/src/View/Job/CompanyView.php
@@ -0,0 +1,22 @@
+ $company->getName(),
+ 'url' => $company->getUrl(),
+ ];
+ }
+}
diff --git a/src/View/Job/DateView.php b/src/View/Job/DateView.php
new file mode 100644
index 0000000..52bc9de
--- /dev/null
+++ b/src/View/Job/DateView.php
@@ -0,0 +1,23 @@
+ (int) $dateTime->format("m"),
+ 'year' => (int) $dateTime->format("Y"),
+ ];
+ }
+}
diff --git a/src/View/JobView.php b/src/View/JobView.php
new file mode 100644
index 0000000..dc510d0
--- /dev/null
+++ b/src/View/JobView.php
@@ -0,0 +1,30 @@
+
+ */
+ public static function create(Job $job): array
+ {
+ return [
+ 'id' => $job->getId(),
+ 'name' => $job->getName(),
+ 'type' => $job->getType(),
+ 'description' => $job->getDescription(),
+ 'company' => CompanyView::create($job->getCompany()),
+ 'from' => DateView::create($job->getFromDate()),
+ 'to' => $job->getToDate() ? DateView::create($job->getToDate()) : null,
+ ];
+ }
+}
diff --git a/src/View/LinkView.php b/src/View/LinkView.php
new file mode 100644
index 0000000..cbef4c6
--- /dev/null
+++ b/src/View/LinkView.php
@@ -0,0 +1,24 @@
+
+ */
+ public static function create(Link $link): array
+ {
+ return [
+ 'id' => $link->getId(),
+ 'title' => $link->getTitle(),
+ 'url' => $link->getUrl(),
+ ];
+ }
+}
diff --git a/src/View/LinksView.php b/src/View/LinksView.php
new file mode 100644
index 0000000..6b67b3a
--- /dev/null
+++ b/src/View/LinksView.php
@@ -0,0 +1,26 @@
+
+ */
+ public static function create(array $links): array
+ {
+ $arr = [];
+
+ foreach ($links as $link) {
+ $arr[] = LinkView::create($link);
+ }
+
+ return $arr;
+ }
+}
diff --git a/src/View/PaginatedView.php b/src/View/PaginatedView.php
new file mode 100644
index 0000000..3a83719
--- /dev/null
+++ b/src/View/PaginatedView.php
@@ -0,0 +1,40 @@
+ $collection
+ * @param callable $view
+ *
+ * @return array{
+ * data: mixed,
+ * pagination: array{
+ * limit: int,
+ * offset: int,
+ * total: int
+ * }
+ * }
+ */
+ public static function create(CollectionChunk $collection, callable $view): array
+ {
+ $data = [];
+ foreach ($collection->chunk as $item) {
+ $data[] = $view($item);
+ }
+
+ return [
+ 'data' => $data,
+ 'pagination' => [
+ 'limit' => $collection->limit,
+ 'offset' => $collection->offset,
+ 'total' => $collection->ofTotalAmount,
+ ],
+ ];
+ }
+}
diff --git a/src/View/ProjectView.php b/src/View/ProjectView.php
new file mode 100644
index 0000000..c2a4c3c
--- /dev/null
+++ b/src/View/ProjectView.php
@@ -0,0 +1,27 @@
+
+ */
+ public static function create(Project $project): array
+ {
+ return [
+ 'id' => $project->getId(),
+ 'name' => $project->getName(),
+ 'image' => ImageView::create($project->getImage()),
+ 'description' => $project->getDescription(),
+ 'links' => LinksView::create($project->getLinks()),
+ 'tags' => TagsView::create($project->getTags()),
+ ];
+ }
+}
diff --git a/src/View/ProjectsView.php b/src/View/ProjectsView.php
new file mode 100644
index 0000000..eb8fdd2
--- /dev/null
+++ b/src/View/ProjectsView.php
@@ -0,0 +1,27 @@
+
+ */
+ public static function create(array $projects): array
+ {
+ $arr = [];
+
+ foreach ($projects as $project) {
+ $arr[] = ProjectView::create($project);
+ }
+
+ return $arr;
+ }
+}
diff --git a/src/View/PropertiesView.php b/src/View/PropertiesView.php
new file mode 100644
index 0000000..c6acc56
--- /dev/null
+++ b/src/View/PropertiesView.php
@@ -0,0 +1,27 @@
+ $collection
+ *
+ * @return array
+ */
+ public static function create(Collection $collection): array
+ {
+ $arr = [];
+
+ foreach ($collection as $item) {
+ $arr[] = PropertyView::create($item);
+ }
+
+ return $arr;
+ }
+}
diff --git a/src/View/PropertyView.php b/src/View/PropertyView.php
new file mode 100644
index 0000000..6e6bc93
--- /dev/null
+++ b/src/View/PropertyView.php
@@ -0,0 +1,23 @@
+
+ */
+ public static function create(Property $property): array
+ {
+ return [
+ 'key' => $property->getKey(),
+ 'value' => $property->getValue(),
+ ];
+ }
+}
diff --git a/src/View/SectionView.php b/src/View/SectionView.php
new file mode 100644
index 0000000..c7a2f07
--- /dev/null
+++ b/src/View/SectionView.php
@@ -0,0 +1,32 @@
+
+ */
+ public static function create(Section $section): array
+ {
+ $skillViews = [];
+
+ foreach ($section->getSkills() as $skill) {
+ $skillViews[] = SkillView::create($skill);
+ }
+
+ return [
+ 'id' => $section->getId(),
+ 'name' => $section->getName(),
+ 'skills' => $skillViews,
+ ];
+ }
+}
diff --git a/src/View/SkillView.php b/src/View/SkillView.php
new file mode 100644
index 0000000..203b120
--- /dev/null
+++ b/src/View/SkillView.php
@@ -0,0 +1,26 @@
+
+ */
+ public static function create(Skill $skill): array
+ {
+ return [
+ 'id' => $skill->getId(),
+ 'name' => $skill->getName(),
+ 'sectionId' => $skill->getSection()->getId(),
+ 'image' => ImageView::create($skill->getImage()),
+ 'description' => $skill->getDescription(),
+ ];
+ }
+}
diff --git a/src/View/TagView.php b/src/View/TagView.php
new file mode 100644
index 0000000..789b73d
--- /dev/null
+++ b/src/View/TagView.php
@@ -0,0 +1,22 @@
+
+ */
+ public static function create(Tag $tag): array
+ {
+ return [
+ 'name' => $tag->getName(),
+ ];
+ }
+}
diff --git a/src/View/TagsView.php b/src/View/TagsView.php
new file mode 100644
index 0000000..47b549b
--- /dev/null
+++ b/src/View/TagsView.php
@@ -0,0 +1,26 @@
+
+ */
+ public static function create(array $tags): array
+ {
+ $arr = [];
+
+ foreach ($tags as $tag) {
+ $arr[] = $tag->getName();
+ }
+
+ return $arr;
+ }
+}
diff --git a/src/View/UserView.php b/src/View/UserView.php
new file mode 100644
index 0000000..b943b95
--- /dev/null
+++ b/src/View/UserView.php
@@ -0,0 +1,25 @@
+
+ */
+ public static function create(UserInterface $user): array
+ {
+ return [
+ 'id' => $user->getId(),
+ 'email' => $user->getEmail(),
+ 'firstName' => $user->getFirstName(),
+ 'lastName' => $user->getLastName(),
+ ];
+ }
+}
diff --git a/symfony.lock b/symfony.lock
new file mode 100644
index 0000000..689cbff
--- /dev/null
+++ b/symfony.lock
@@ -0,0 +1,179 @@
+{
+ "doctrine/doctrine-bundle": {
+ "version": "2.8",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "2.8",
+ "ref": "719d615af3150a59a02acc97c285d689e5dcf43f"
+ },
+ "files": [
+ "config/packages/doctrine.yaml",
+ "src/Entity/.gitignore",
+ "src/Repository/.gitignore"
+ ]
+ },
+ "doctrine/doctrine-migrations-bundle": {
+ "version": "3.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "3.1",
+ "ref": "1d01ec03c6ecbd67c3375c5478c9a423ae5d6a33"
+ },
+ "files": [
+ "config/packages/doctrine_migrations.yaml",
+ "migrations/.gitignore"
+ ]
+ },
+ "phpunit/phpunit": {
+ "version": "10.0",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "9.3",
+ "ref": "a6249a6c4392e9169b87abf93225f7f9f59025e6"
+ },
+ "files": [
+ ".env.test",
+ "phpunit.xml.dist",
+ "tests/bootstrap.php"
+ ]
+ },
+ "ser/dto-request-bundle": {
+ "version": "dev-master"
+ },
+ "squizlabs/php_codesniffer": {
+ "version": "3.7",
+ "recipe": {
+ "repo": "github.com/symfony/recipes-contrib",
+ "branch": "main",
+ "version": "3.6",
+ "ref": "1019e5c08d4821cb9b77f4891f8e9c31ff20ac6f"
+ },
+ "files": [
+ "phpcs.xml.dist"
+ ]
+ },
+ "symfony/console": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "5.3",
+ "ref": "da0c8be8157600ad34f10ff0c9cc91232522e047"
+ },
+ "files": [
+ "bin/console"
+ ]
+ },
+ "symfony/flex": {
+ "version": "2.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "1.0",
+ "ref": "146251ae39e06a95be0fe3d13c807bcf3938b172"
+ },
+ "files": [
+ ".env"
+ ]
+ },
+ "symfony/framework-bundle": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "6.2",
+ "ref": "af47254c5e4cd543e6af3e4508298ffebbdaddd3"
+ },
+ "files": [
+ "config/packages/cache.yaml",
+ "config/packages/framework.yaml",
+ "config/preload.php",
+ "config/routes/framework.yaml",
+ "config/services.yaml",
+ "public/index.php",
+ "src/Controller/.gitignore",
+ "src/Kernel.php"
+ ]
+ },
+ "symfony/phpunit-bridge": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "5.3",
+ "ref": "819d3d2ffa4590eba0b8f4f3e5e89415ee4e45c3"
+ },
+ "files": [
+ ".env.test",
+ "bin/phpunit",
+ "phpunit.xml.dist",
+ "tests/bootstrap.php"
+ ]
+ },
+ "symfony/routing": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "6.2",
+ "ref": "e0a11b4ccb8c9e70b574ff5ad3dfdcd41dec5aa6"
+ },
+ "files": [
+ "config/packages/routing.yaml",
+ "config/routes.yaml"
+ ]
+ },
+ "symfony/security-bundle": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "6.0",
+ "ref": "8a5b112826f7d3d5b07027f93786ae11a1c7de48"
+ },
+ "files": [
+ "config/packages/security.yaml"
+ ]
+ },
+ "symfony/twig-bundle": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "5.4",
+ "ref": "bb2178c57eee79e6be0b297aa96fc0c0def81387"
+ },
+ "files": [
+ "config/packages/twig.yaml",
+ "templates/base.html.twig"
+ ]
+ },
+ "symfony/validator": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "5.3",
+ "ref": "c32cfd98f714894c4f128bb99aa2530c1227603c"
+ },
+ "files": [
+ "config/packages/validator.yaml"
+ ]
+ },
+ "symfony/web-profiler-bundle": {
+ "version": "6.2",
+ "recipe": {
+ "repo": "github.com/symfony/recipes",
+ "branch": "main",
+ "version": "6.1",
+ "ref": "e42b3f0177df239add25373083a564e5ead4e13a"
+ },
+ "files": [
+ "config/packages/web_profiler.yaml",
+ "config/routes/web_profiler.yaml"
+ ]
+ }
+}
diff --git a/tests/Controller/Job/CreateControllerTest.php b/tests/Controller/Job/CreateControllerTest.php
new file mode 100644
index 0000000..eefe149
--- /dev/null
+++ b/tests/Controller/Job/CreateControllerTest.php
@@ -0,0 +1,53 @@
+sendRequest('POST', self::API_URL);
+
+ self::assertEquals(
+ '{"code":400,"message":"Request body is empty"}',
+ $response->getContent()
+ );
+ }
+
+ public function testAccessForbidden(): void
+ {
+ $response = $this->sendRequest('POST', self::API_URL, [
+ 'name' => 'name',
+ 'company' => [
+ 'name' => 'company.name',
+ 'url' => 'company.url',
+ ],
+ 'description' => 'description',
+ 'type' => 'type',
+ 'from' => [
+ 'month' => 1,
+ 'year' => 2011,
+ ],
+ 'to' => [
+ 'month' => 2,
+ 'year' => 2012,
+ ],
+ ]);
+
+ self::assertEquals(
+ '{"code":403,"message":"You\u0027re not allowed to create jobs"}',
+ $response->getContent()
+ );
+ }
+}
diff --git a/tests/Controller/Job/DeleteControllerTest.php b/tests/Controller/Job/DeleteControllerTest.php
new file mode 100644
index 0000000..c061493
--- /dev/null
+++ b/tests/Controller/Job/DeleteControllerTest.php
@@ -0,0 +1,12 @@
+browser = self::createClient();
+
+ $this->initDatabase(self::$kernel);
+ }
+
+ protected function tearDown(): void
+ {
+ parent::tearDown();
+ }
+
+ /**
+ * @param string $username
+ * @param string $password
+ *
+ * @return void
+ *
+ * @throws \Exception
+ */
+ public function login(string $username, string $password): void
+ {
+ $userRepository = self::getContainer()->get(UserRepository::class);
+
+ $user = $userRepository->findOneByEmail($username);
+
+ if ($user === null) {
+ return;
+ }
+
+ $this->browser->loginUser($user);
+ }
+
+ /**
+ * @param string $method
+ * @param string $uri
+ * @param array $parameters
+ *
+ * @return Response
+ */
+ final protected function sendRequest(string $method, string $uri, array $parameters = []): Response
+ {
+ $this->browser->jsonRequest($method, $uri, $parameters);
+
+ return $this->browser->getResponse();
+ }
+
+
+ final protected function getEntityManager(): EntityManagerInterface
+ {
+ return $this->entityManager;
+ }
+
+ private function initDatabase(KernelInterface $kernel)
+ {
+ $this->entityManager = self::$kernel->getContainer()
+ ->get('doctrine')
+ ->getManager();
+
+ $metaData = $this->entityManager->getMetadataFactory()->getAllMetadata();
+ $schemaTool = new SchemaTool($this->entityManager);
+ $schemaTool->updateSchema($metaData);
+ }
+}
diff --git a/tests/Repository/DataFixtures/LoadProjectsData.php b/tests/Repository/DataFixtures/LoadProjectsData.php
new file mode 100644
index 0000000..f6e83bc
--- /dev/null
+++ b/tests/Repository/DataFixtures/LoadProjectsData.php
@@ -0,0 +1,10 @@
+repository = new EducationRepository($this->getEntityManager());
+ }
+
+ public function testSaveEducation(): void
+ {
+ $education = new Education(
+ 'institution',
+ 'faculty',
+ 'specialization',
+ new DateTime('20-01-2020'),
+ new DateTime('20-03-2020'),
+ );
+
+ $this->repository->save($education);
+
+ $savedEducation = $this->repository->findOneById($education->getId());
+
+ $this->assertEquals($education, $savedEducation);
+ }
+}
diff --git a/tests/Repository/JobRepositoryTest.php b/tests/Repository/JobRepositoryTest.php
new file mode 100644
index 0000000..7d688ef
--- /dev/null
+++ b/tests/Repository/JobRepositoryTest.php
@@ -0,0 +1,41 @@
+repository = new JobRepository($this->getEntityManager());
+ }
+
+ public function testSaveJob(): void
+ {
+ $job = new Job(
+ 'job name',
+ 'type',
+ 'some description',
+ new Company('name', 'url-to-company'),
+ new DateTime('01-01-2001'),
+ new DateTime('01-01-2002'),
+ );
+
+ $this->repository->save($job);
+
+ $savedJob = $this->repository->findOneById($job->getId());
+
+ $this->assertEquals($job, $savedJob);
+ }
+}
diff --git a/tests/Repository/LinkRepositoryTest.php b/tests/Repository/LinkRepositoryTest.php
new file mode 100644
index 0000000..1368df0
--- /dev/null
+++ b/tests/Repository/LinkRepositoryTest.php
@@ -0,0 +1,26 @@
+repository = new LinkRepository($this->getEntityManager());
+ }
+
+ public function testAddLink(): void
+ {
+
+ }
+}
diff --git a/tests/Repository/ProjectRepositoryTest.php b/tests/Repository/ProjectRepositoryTest.php
new file mode 100644
index 0000000..669959d
--- /dev/null
+++ b/tests/Repository/ProjectRepositoryTest.php
@@ -0,0 +1,42 @@
+repository = new ProjectRepository($this->getEntityManager());
+ }
+
+ public function testSaveProject(): void
+ {
+ $project = new Project(
+ 'job name',
+ 'type',
+ new File(
+ 'somefile.jpeg',
+ 'path/to/file/',
+ 'image/png',
+ 300,
+ ),
+ );
+
+ $this->repository->save($project);
+
+ $savedProject = $this->repository->findOneById($project->getId());
+
+ $this->assertEquals($project, $savedProject);
+ }
+}
diff --git a/tests/Repository/PropertyRepositoryTest.php b/tests/Repository/PropertyRepositoryTest.php
new file mode 100644
index 0000000..cccbedd
--- /dev/null
+++ b/tests/Repository/PropertyRepositoryTest.php
@@ -0,0 +1,36 @@
+repository = new PropertyRepository($this->getEntityManager());
+ }
+
+ public function testSaveProperty(): void
+ {
+ $propertyKey = 'some_property_key';
+ $property = new Property(
+ $propertyKey,
+ 'value',
+ );
+
+ $this->repository->save($property);
+
+ $savedProperty = $this->repository->findByKey($propertyKey);
+
+ $this->assertEquals($property, $savedProperty);
+ }
+}
diff --git a/tests/Repository/SectionRepositoryTest.php b/tests/Repository/SectionRepositoryTest.php
new file mode 100644
index 0000000..b450572
--- /dev/null
+++ b/tests/Repository/SectionRepositoryTest.php
@@ -0,0 +1,34 @@
+repository = new SectionRepository($this->getEntityManager());
+ }
+
+ public function testSaveSection(): void
+ {
+ $section = new Section(
+ 'section name'
+ );
+
+ $this->repository->save($section);
+
+ $savedSection = $this->repository->findOneById($section->getId());
+
+ $this->assertEquals($section, $savedSection);
+ }
+}
diff --git a/tests/Repository/SkillRepositoryTest.php b/tests/Repository/SkillRepositoryTest.php
new file mode 100644
index 0000000..ce5627d
--- /dev/null
+++ b/tests/Repository/SkillRepositoryTest.php
@@ -0,0 +1,52 @@
+repository = new SkillRepository($this->getEntityManager());
+ }
+
+ public function testSaveSkill(): void
+ {
+ $em = $this->getEntityManager();
+
+ $file = new File(
+ 'file.ext',
+ '/path/to/file/',
+ 'mime/type',
+ 100,
+ );
+ $section = new Section('section name');
+
+ $em->persist($file);
+ $em->persist($section);
+
+ $skill = new Skill(
+ 'skill name',
+ $section,
+ $file,
+ 'description',
+ );
+
+ $this->repository->save($skill);
+
+ $savedSkill = $this->repository->findOneById($skill->getId());
+
+ $this->assertEquals($skill, $savedSkill);
+ }
+}
diff --git a/tests/Repository/StorageRepositoryTest.php b/tests/Repository/StorageRepositoryTest.php
new file mode 100644
index 0000000..01eb33b
--- /dev/null
+++ b/tests/Repository/StorageRepositoryTest.php
@@ -0,0 +1,37 @@
+repository = new StorageRepository($this->getEntityManager());
+ }
+
+ public function testSaveFile(): void
+ {
+ $file = new File(
+ 'file.ext',
+ '/path/to/file/',
+ 'mime/type',
+ 100,
+ );
+
+ $this->repository->save($file);
+
+ $savedFile = $this->repository->findOneById($file->getId());
+
+ $this->assertEquals($file, $savedFile);
+ }
+}
diff --git a/tests/Repository/TagRepositoryTest.php b/tests/Repository/TagRepositoryTest.php
new file mode 100644
index 0000000..d336653
--- /dev/null
+++ b/tests/Repository/TagRepositoryTest.php
@@ -0,0 +1,41 @@
+repository = new TagRepository($this->getEntityManager());
+ }
+
+ public function testSaveTag(): void
+ {
+ $tagNames = [
+ 'tag1',
+ 'TAG2',
+ ];
+
+ $tags = [];
+
+ foreach ($tagNames as $tagName){
+ $tags[] = new Tag($tagName);
+ }
+
+ $this->repository->saveTags($tags);
+
+ $savedTags = $this->repository->findTagsByNames($tagNames);
+
+ $this->assertSame($savedTags, $tags);
+ }
+}
diff --git a/tests/Repository/UserRepositoryTest.php b/tests/Repository/UserRepositoryTest.php
new file mode 100644
index 0000000..55a69a2
--- /dev/null
+++ b/tests/Repository/UserRepositoryTest.php
@@ -0,0 +1,38 @@
+repository = new UserRepository($this->getEntityManager());
+ }
+
+ public function testSaveUser(): void
+ {
+ $expectedEmail = 'test@user.com';
+ $user = new User(
+ $expectedEmail,
+ 'pwd',
+ 'first name',
+ 'last name',
+ );
+
+ $this->repository->save($user);
+
+ $savedUser = $this->repository->findOneByEmail($expectedEmail);
+
+ $this->assertEquals($user, $savedUser);
+ }
+}
diff --git a/tests/Service/Education/EducationServiceTest.php b/tests/Service/Education/EducationServiceTest.php
new file mode 100644
index 0000000..14346ab
--- /dev/null
+++ b/tests/Service/Education/EducationServiceTest.php
@@ -0,0 +1,260 @@
+educationRepository = $this->createMock(EducationRepositoryInterface::class);
+ $this->validationService = $this->createMock(ValidationServiceInterface::class);
+
+ $this->service = new EducationService(
+ $this->educationRepository,
+ $this->validationService,
+ );
+
+ $this->testData = new EducationData(
+ institution: 'institution',
+ faculty: 'faculty',
+ specialization: 'specialization',
+ from: new DateData(1, 2010),
+ to: new DateData(1, 2010),
+ );
+ }
+
+ public function testAddingEducationWithInvalidData(): void
+ {
+ $expectedError = new ValidationError("some error", []);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn($expectedError);
+
+ $result = $this->service->addEducation($this->testData);
+
+ $this->assertInstanceOf(ErrorInterface::class, $result);
+ $this->assertSame($expectedError, $result);
+ }
+
+ public function testEducationSaving(): void
+ {
+ $this->testData->to = new DateData(1, 2010);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn(null);
+
+ $expectedEducation = null;
+ $this->educationRepository
+ ->expects($this->once())
+ ->method('save')
+ ->with($this->callback(function (Education $education) use (&$expectedEducation): bool {
+ $this->assertEquals($this->testData->institution, $education->getInstitution());
+ $this->assertEquals($this->testData->faculty, $education->getFaculty());
+ $this->assertEquals($this->testData->specialization, $education->getSpecialization());
+ $this->assertEquals($this->testData->from->month, $education->getFromDate()->format('m'));
+ $this->assertEquals($this->testData->from->year, $education->getFromDate()->format('Y'));
+
+ $expectedEducation = $education;
+
+ return true;
+ }));
+
+ $result = $this->service->addEducation($this->testData);
+
+ $this->assertInstanceOf(Education::class, $result);
+ $this->assertsame($expectedEducation, $result);
+ }
+
+ public function testUpdatingEducationWithInvalidData(): void
+ {
+ $educationEntity = new Education(
+ 'institution',
+ 'faculty',
+ 'specialization',
+ new DateTime('now'),
+ new DateTime('now'),
+ );
+
+ $expectedError = new ValidationError("some error", []);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn($expectedError);
+
+ $result = $this->service->updateEducation($educationEntity, $this->testData);
+
+ $this->assertInstanceOf(ErrorInterface::class, $result);
+ $this->assertSame($expectedError, $result);
+ }
+
+ public function testEducationUpdating(): void
+ {
+ $educationEntity = new Education(
+ 'institution',
+ 'faculty',
+ 'specialization',
+ new DateTime('now'),
+ new DateTime('now'),
+ );
+
+ $this->testData->to = new DateData(1, 2010);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn(null);
+
+ $this->educationRepository
+ ->expects($this->once())
+ ->method('save')
+ ->with($this->callback(function (Education $education) use (&$expectedEducation): bool {
+ $this->assertEquals($this->testData->institution, $education->getInstitution());
+ $this->assertEquals($this->testData->faculty, $education->getFaculty());
+ $this->assertEquals($this->testData->specialization, $education->getSpecialization());
+ $this->assertEquals($this->testData->from->month, $education->getFromDate()->format('m'));
+ $this->assertEquals($this->testData->from->year, $education->getFromDate()->format('Y'));
+
+ $expectedEducation = $education;
+
+ return true;
+ }));
+
+ $result = $this->service->updateEducation($educationEntity, $this->testData);
+
+ $this->assertInstanceOf(Education::class, $result);
+ $this->assertsame($educationEntity, $result);
+ }
+
+ public function testGetNotExistsEducationById(): void
+ {
+ $educationId = 123;
+
+ $this->educationRepository
+ ->expects($this->once())
+ ->method('findOneById')
+ ->with($this->identicalTo($educationId))
+ ->willReturn(null);
+
+ $education = $this->service->getEducationById($educationId);
+
+ $this->assertNull($education);
+ }
+ public function testGetEducationById(): void
+ {
+ $educationId = 123;
+
+ $expectedEducation = $this->createMock(Education::class);
+ $this->educationRepository
+ ->expects($this->once())
+ ->method('findOneById')
+ ->with($this->identicalTo($educationId))
+ ->willReturn($expectedEducation);
+
+ $education = $this->service->getEducationById($educationId);
+
+ $this->assertEquals($expectedEducation, $education);
+ }
+
+ public function testDeleteEducation(): void
+ {
+ $educationEntity = new Education(
+ 'institution',
+ 'faculty',
+ 'specialization',
+ new DateTime('now'),
+ new DateTime('now'),
+ );
+
+ $this->educationRepository
+ ->expects($this->once())
+ ->method('delete')
+ ->with($this->callback(function (Education $education) use (&$educationEntity): bool {
+ $this->assertEquals($educationEntity, $education);
+
+ return true;
+ }));
+
+ $this->service->deleteEducation($educationEntity);
+ }
+
+ public function testGetEducationsCount(): void
+ {
+ $expectedCount = 123;
+
+ $filter = new EducationFilter(10, 0);
+
+ $this->educationRepository
+ ->expects($this->once())
+ ->method('getCount')
+ ->with($this->identicalTo($filter))
+ ->willReturn($expectedCount);
+
+ $result = $this->service->getCount($filter);
+
+ $this->assertEquals($expectedCount, $result);
+ }
+
+ public function testGetEducations(): void
+ {
+ $expectedCollection = $this->createMock(Collection::class);
+
+ $filter = new EducationFilter(10, 0);
+ $sort = new EducationSort('id', 'asc');
+
+ $this->educationRepository
+ ->expects($this->once())
+ ->method('getEducations')
+ ->with($this->identicalTo($filter))
+ ->willReturn($expectedCollection);
+
+ $result = $this->service->getEducations($filter, $sort);
+
+ $this->assertEquals($expectedCollection, $result);
+ }
+}
diff --git a/tests/Service/Job/JobServiceTest.php b/tests/Service/Job/JobServiceTest.php
new file mode 100644
index 0000000..a141be0
--- /dev/null
+++ b/tests/Service/Job/JobServiceTest.php
@@ -0,0 +1,270 @@
+jobRepository = $this->createMock(JobRepositoryInterface::class);
+ $this->validationService = $this->createMock(ValidationServiceInterface::class);
+
+ $this->service = new JobService(
+ $this->jobRepository,
+ $this->validationService,
+ );
+
+ $this->testData = new JobData(
+ name: 'name',
+ description: 'description',
+ type:'type of work',
+ company: new CompanyData(
+ 'company.name',
+ 'company.url'
+ ),
+ from: new DateData(1, 2010),
+ );
+ }
+
+ public function testAddingJobWithInvalidData(): void
+ {
+ $expectedError = new ValidationError("some error", []);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn($expectedError);
+
+ $result = $this->service->addJob($this->testData);
+
+ $this->assertInstanceOf(ErrorInterface::class, $result);
+ $this->assertSame($expectedError, $result);
+ }
+
+ public function testJobSaving(): void
+ {
+ $this->testData->to = new DateData(1, 2010);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn(null);
+
+ $expectedJob = null;
+ $this->jobRepository
+ ->expects($this->once())
+ ->method('save')
+ ->with($this->callback(function (Job $job) use (&$expectedJob): bool {
+ $this->assertEquals($this->testData->name, $job->getName());
+ $this->assertEquals($this->testData->type, $job->getType());
+ $this->assertEquals($this->testData->description, $job->getDescription());
+ $this->assertEquals($this->testData->company->name, $job->getCompany()->getName());
+ $this->assertEquals($this->testData->company->url, $job->getCompany()->getUrl());
+ $this->assertEquals($this->testData->from->month, $job->getFromDate()->format('m'));
+ $this->assertEquals($this->testData->from->year, $job->getFromDate()->format('Y'));
+
+ $expectedJob = $job;
+
+ return true;
+ }));
+
+ $result = $this->service->addJob($this->testData);
+
+ $this->assertInstanceOf(Job::class, $result);
+ $this->assertsame($expectedJob, $result);
+ }
+
+ public function testUpdatingJobWithInvalidData(): void
+ {
+ $jobEntity = new Job(
+ 'job name',
+ 'job type',
+ 'job description',
+ new Company('job company name', 'job company url'),
+ new DateTime('now'),
+ new DateTime('now'),
+ );
+
+ $expectedError = new ValidationError("some error", []);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn($expectedError);
+
+ $result = $this->service->updateJob($jobEntity, $this->testData);
+
+ $this->assertInstanceOf(ErrorInterface::class, $result);
+ $this->assertSame($expectedError, $result);
+ }
+
+ public function testJobUpdating(): void
+ {
+ $jobEntity = new Job(
+ 'job name',
+ 'job type',
+ 'job description',
+ new Company('job company name', 'job company url'),
+ new DateTime('now'),
+ new DateTime('now'),
+ );
+
+ $this->testData->to = new DateData(1, 2010);
+
+ $this->validationService
+ ->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($this->testData))
+ ->willReturn(null);
+
+ $this->jobRepository
+ ->expects($this->once())
+ ->method('save')
+ ->with($this->callback(function (Job $job): bool {
+ $this->assertEquals($this->testData->name, $job->getName());
+ $this->assertEquals($this->testData->type, $job->getType());
+ $this->assertEquals($this->testData->description, $job->getDescription());
+ $this->assertEquals($this->testData->company->name, $job->getCompany()->getName());
+ $this->assertEquals($this->testData->company->url, $job->getCompany()->getUrl());
+ $this->assertEquals($this->testData->from->month, $job->getFromDate()->format('m'));
+ $this->assertEquals($this->testData->from->year, $job->getFromDate()->format('Y'));
+
+ return true;
+ }));
+
+ $result = $this->service->updateJob($jobEntity, $this->testData);
+
+ $this->assertInstanceOf(Job::class, $result);
+ $this->assertsame($jobEntity, $result);
+ }
+
+ public function testGetNotExistsJobById(): void
+ {
+ $jobId = 123;
+
+ $this->jobRepository
+ ->expects($this->once())
+ ->method('findOneById')
+ ->with($this->identicalTo($jobId))
+ ->willReturn(null);
+
+ $job = $this->service->getJobById($jobId);
+
+ $this->assertNull($job);
+ }
+ public function testGetJobById(): void
+ {
+ $jobId = 123;
+
+ $expectedJob = $this->createMock(Job::class);
+ $this->jobRepository
+ ->expects($this->once())
+ ->method('findOneById')
+ ->with($this->identicalTo($jobId))
+ ->willReturn($expectedJob);
+
+ $job = $this->service->getJobById($jobId);
+
+ $this->assertEquals($expectedJob, $job);
+ }
+
+ public function testDeleteJob(): void
+ {
+ $jobEntity = new Job(
+ 'job name',
+ 'job type',
+ 'job description',
+ new Company('job company name', 'job company url'),
+ new DateTime('now'),
+ new DateTime('now'),
+ );
+
+ $this->jobRepository
+ ->expects($this->once())
+ ->method('delete')
+ ->with($this->callback(function (Job $job) use (&$jobEntity): bool {
+ $this->assertEquals($jobEntity, $job);
+
+ return true;
+ }));
+
+ $this->service->deleteJob($jobEntity);
+ }
+
+ public function testGetJobsCount(): void
+ {
+ $expectedCount = 123;
+
+ $filter = new JobFilter(10, 0);
+
+ $this->jobRepository
+ ->expects($this->once())
+ ->method('getCount')
+ ->with($this->identicalTo($filter))
+ ->willReturn($expectedCount);
+
+ $result = $this->service->getCount($filter);
+
+ $this->assertEquals($expectedCount, $result);
+ }
+
+ public function testGetJobs(): void
+ {
+ $expectedCollection = $this->createMock(Collection::class);
+
+ $filter = new JobFilter(10, 0);
+ $sort = new JobSort('id', 'asc');
+
+ $this->jobRepository
+ ->expects($this->once())
+ ->method('getJobs')
+ ->with($this->identicalTo($filter))
+ ->willReturn($expectedCollection);
+
+ $result = $this->service->getJobs($filter, $sort);
+
+ $this->assertEquals($expectedCollection, $result);
+ }
+}
diff --git a/tests/Service/Response/ResponseFactoryTest.php b/tests/Service/Response/ResponseFactoryTest.php
new file mode 100644
index 0000000..71b1b83
--- /dev/null
+++ b/tests/Service/Response/ResponseFactoryTest.php
@@ -0,0 +1,98 @@
+responseFactory = new ResponseFactory();
+ }
+
+ public function testCreateResponse(): void
+ {
+ $dataToConvert = ['a' => "one", "b" => "two"];
+ $expectedData = json_encode($dataToConvert);
+ $expectedCode = 200;
+ $expectedHeadersKey = "Authorization";
+ $expectedHeadersValue = "Bearer someToken";
+
+ $response = $this
+ ->responseFactory
+ ->createResponse(
+ $dataToConvert,
+ $expectedCode,
+ [$expectedHeadersKey => $expectedHeadersValue]
+ );
+
+ $this->assertEquals($expectedData, $response->getContent());
+ $this->assertEquals($expectedCode, $response->getStatusCode());
+ $this->assertTrue(
+ $response->headers->contains($expectedHeadersKey, $expectedHeadersValue),
+ "Headers not contains expected header"
+ );
+ }
+
+ public function testForbiddenResponse(): void
+ {
+ $expectedCode = Response::HTTP_FORBIDDEN;
+ $message = "Access forbidden for your role";
+ $expectedResponse = json_encode(['code' => $expectedCode, 'message' => $message]);
+
+ $response = $this
+ ->responseFactory
+ ->forbidden($message);
+
+ $this->assertEquals($expectedResponse, $response->getContent());
+ $this->assertEquals($expectedCode, $response->getStatusCode());
+ }
+
+ public function testNotFoundResponse(): void
+ {
+ $expectedCode = Response::HTTP_NOT_FOUND;
+ $message = "Page not found";
+ $expectedResponse = json_encode(['code' => $expectedCode, 'message' => $message]);
+
+ $response = $this
+ ->responseFactory
+ ->notFound($message);
+
+ $this->assertEquals($expectedResponse, $response->getContent());
+ $this->assertEquals($expectedCode, $response->getStatusCode());
+ }
+
+ public function testBadRequestResponse(): void
+ {
+ $expectedCode = Response::HTTP_BAD_REQUEST;
+ $message = "Bad request";
+ $expectedResponse = json_encode(['code' => $expectedCode, 'message' => $message]);
+
+ $response = $this
+ ->responseFactory
+ ->badRequest($message);
+
+ $this->assertEquals($expectedResponse, $response->getContent());
+ $this->assertEquals($expectedCode, $response->getStatusCode());
+ }
+
+ public function testUnauthorizedResponse(): void
+ {
+ $expectedCode = Response::HTTP_UNAUTHORIZED;
+ $message = "unauthorized user";
+ $expectedResponse = json_encode(['code' => $expectedCode, 'message' => $message]);
+
+ $response = $this
+ ->responseFactory
+ ->unauthorized($message);
+
+ $this->assertEquals($expectedResponse, $response->getContent());
+ $this->assertEquals($expectedCode, $response->getStatusCode());
+ }
+}
diff --git a/tests/Service/Validation/ValidationServiceTest.php b/tests/Service/Validation/ValidationServiceTest.php
new file mode 100644
index 0000000..4848f2f
--- /dev/null
+++ b/tests/Service/Validation/ValidationServiceTest.php
@@ -0,0 +1,66 @@
+validator = $this->createMock(ValidatorInterface::class);
+
+ $this->service = new ValidationService($this->validator);
+ }
+
+ public function testValidateIfDataInvalid(): void
+ {
+ $expectedErrorKey = 'some.path';
+ $expectedErrorValue = 'some error';
+
+ $violation = $this->createMock(ConstraintViolationInterface::class);
+
+ $violation->expects($this->once())
+ ->method('getMessage')
+ ->willReturn($expectedErrorValue);
+
+ $violation->expects($this->once())
+ ->method('getPropertyPath')
+ ->willReturn($expectedErrorKey);
+
+ $violations = new ConstraintViolationList([$violation]);
+
+ $someData = [];
+ $this->validator->expects($this->once())
+ ->method('validate')
+ ->with($this->identicalTo($someData))
+ ->willReturn($violations);
+
+ $error = $this->service->validate($someData);
+
+ $this->assertInstanceOf(ErrorInterface::class, $error);
+ $this->assertEquals('Validation Error', $error->getMessage());
+ $this->assertEquals([$expectedErrorKey => $expectedErrorValue], $error->getDetails());
+ }
+}
diff --git a/tests/bootstrap.php b/tests/bootstrap.php
new file mode 100644
index 0000000..1e75e32
--- /dev/null
+++ b/tests/bootstrap.php
@@ -0,0 +1,15 @@
+bootEnv(dirname(__DIR__) . '/.env');
+}
+
+if ($_SERVER['APP_DEBUG']) {
+ umask(0000);
+}