From b5e824ba6e8ef461fb80eb2717b930cf2e5dcc5c Mon Sep 17 00:00:00 2001 From: Robert Zondervan Date: Fri, 8 Nov 2024 09:51:51 +0100 Subject: [PATCH 01/18] Fixes on object validation --- lib/Service/ObjectService.php | 6 +++--- lib/Service/ValidationService.php | 4 ++++ 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/lib/Service/ObjectService.php b/lib/Service/ObjectService.php index 9a5e022f..e0befbbf 100644 --- a/lib/Service/ObjectService.php +++ b/lib/Service/ObjectService.php @@ -304,13 +304,13 @@ public function getAllObjects(string $objectType, ?int $limit = null, ?int $offs */ public function saveObject(string $objectType, array $object, bool $updateVersion = true): mixed { - // Get the appropriate mapper for the object type - $mapper = $this->getMapper($objectType); - if ($objectType === 'publication') { $object = $this->validationService->validatePublication($object); } + // Get the appropriate mapper for the object type + $mapper = $this->getMapper($objectType); + // If the object has an id, update it; otherwise, create a new object if (isset($object['id']) === true) { return $mapper->updateFromArray($object['id'], $object, $updateVersion); diff --git a/lib/Service/ValidationService.php b/lib/Service/ValidationService.php index dc863d36..38371705 100644 --- a/lib/Service/ValidationService.php +++ b/lib/Service/ValidationService.php @@ -52,6 +52,10 @@ public function validatePublication(array $publication): array $validator = new Validator(); $validator->setMaxErrors(100); + if(empty($publicationType->getProperties()) === true) { + return $publication; + } + $result = $validator->validate(data: (object) json_decode(json_encode($publication['data'])), schema: $publicationType->getSchema($this->urlGenerator)); $publication['validation'] = []; From 07918dbc72202111d47be9f2b4ebd83e100c3fec Mon Sep 17 00:00:00 2001 From: Robert Zondervan Date: Fri, 8 Nov 2024 09:55:43 +0100 Subject: [PATCH 02/18] Recover from revert on validationService --- lib/Service/ValidationService.php | 113 ++++++++---------------------- 1 file changed, 28 insertions(+), 85 deletions(-) diff --git a/lib/Service/ValidationService.php b/lib/Service/ValidationService.php index 0c15c957..38371705 100644 --- a/lib/Service/ValidationService.php +++ b/lib/Service/ValidationService.php @@ -3,9 +3,18 @@ namespace OCA\OpenCatalogi\Service; use OCA\OpenCatalogi\Db\CatalogMapper; +use OCA\OpenCatalogi\Db\Publication; +use OCA\OpenCatalogi\Db\PublicationType; +use OCP\AppFramework\Db\DoesNotExistException; +use OCP\AppFramework\Db\MultipleObjectsReturnedException; use OCP\AppFramework\OCS\OCSBadRequestException; use OCP\AppFramework\OCS\OCSNotFoundException; use OCP\IAppConfig; +use OCP\IURLGenerator; +use Opis\JsonSchema\Errors\ErrorFormatter; +use Opis\JsonSchema\Validator; +use Psr\Container\ContainerExceptionInterface; +use Psr\Container\NotFoundExceptionInterface; /** * Class ValidationService @@ -14,112 +23,46 @@ */ class ValidationService { - /** - * @var string The name of the application. - */ - private string $appName; - - /** - * @var array The current MongoDB Config. - */ - private array $mongodbConfig; - /** - * ValidationService constructor. - * - * @param IAppConfig $config The application config - * @param CatalogMapper $catalogMapper The catalog mapper. - * @param ObjectService $objectService The object service. - */ public function __construct( - private readonly IAppConfig $config, - private readonly CatalogMapper $catalogMapper, private readonly ObjectService $objectService, - ) { - $this->appName = 'opencatalogi'; - - // Initialize MongoDB configuration - $this->mongodbConfig = [ - 'base_uri' => $this->config->getValueString(app: $this->appName, key: 'mongodbLocation'), - 'headers' => ['api-key' => $this->config->getValueString(app: $this->appName, key: 'mongodbKey')], - 'mongodbCluster' => $this->config->getValueString(app: $this->appName, key:'mongodbCluster') - ]; - } - - /** - * Get the MongoDB configuration. - * - * @return array The mongodb config. - */ - public function getMongodbConfig(): array + private readonly IURLGenerator $urlGenerator, + ) { - return $this->mongodbConfig; } /** - * Fetches a catalog from either the local database or mongodb + * Validate a publication to the definition defined in the PublicationType. * - * @param string $id The id of the catalog to be fetched. - * @return array The JSON Serialised catalog. + * @param Publication $publication The publication to validate. + * @return Publication The validated publication. * - * @throws OCSNotFoundException If the catalog is not found. - * @throws \GuzzleHttp\Exception\GuzzleException + * @throws DoesNotExistException + * @throws MultipleObjectsReturnedException + * @throws ContainerExceptionInterface + * @throws NotFoundExceptionInterface */ - public function getCatalog (string $id): array + public function validatePublication(array $publication): array { - // Check if MongoDB storage is enabled - if ($this->config->hasKey(app: $this->appName, key: 'mongoStorage') !== false - && $this->config->getValueString(app: $this->appName, key: 'mongoStorage') === '1' - ) { - $filter = ['id' => $id, '_schema' => 'catalog']; + $publicationTypeId = $publication['publicationType']; + $publicationType = $this->objectService->getObject(objectType: 'publicationType', id: $publicationTypeId); - try { - return $this->objectService->findObject(filters: $filter, config: $this->getMongodbConfig()); - } catch (OCSNotFoundException $exception) { - throw new OCSNotFoundException(message: 'Catalog not found for id: ' . $id); - } - } + $publicationType = (new PublicationType())->hydrate($publicationType); - // If MongoDB storage is not enabled, fetch from local database - return $this->catalogMapper->find(id: $id)->jsonSerialize(); - } + $validator = new Validator(); + $validator->setMaxErrors(100); if(empty($publicationType->getProperties()) === true) { return $publication; } $result = $validator->validate(data: (object) json_decode(json_encode($publication['data'])), schema: $publicationType->getSchema($this->urlGenerator)); - /** - * Validates a publication against the rules set for the publication. - * - * @param array $publication The publication to be validated. - * @return array The publication after it has been validated. - * - * @throws OCSBadRequestException Thrown if the object does not validate - * @throws OCSNotFoundException Thrown if the catalog is not found - */ - public function validatePublication(array $publication): array - { - // Check for required fields - $requiredFields = ['catalogi', 'publicationType']; - foreach ($requiredFields as $field) { - if (isset($publication[$field]) === false) { - throw new OCSBadRequestException(message: $field . ' is required but not given.'); - } - } - - $catalog = $publication['catalogi']; - $publicationType = $publication['publicationType']; - try { - $catalog = $this->getCatalog($catalog); - } catch (OCSNotFoundException $exception) { - throw new OCSNotFoundException(message: $exception->getMessage()); - } + $publication['validation'] = []; - // Check if the given publicationType is present in the catalog - if (in_array(needle: $publicationType, haystack: $catalog['publicationType']) === false) { - throw new OCSBadRequestException(message: 'Given publicationType object not present in catalog'); + if ($result->hasError()) { + $errors = (new ErrorFormatter())->format($result->error()); + $publication['validation'] = $errors; } return $publication; From 149387ace8a89335f3aef646b281fa39fb2d6bfb Mon Sep 17 00:00:00 2001 From: Robert Zondervan Date: Fri, 8 Nov 2024 09:56:24 +0100 Subject: [PATCH 03/18] Recover from revert on objectService --- lib/Service/ObjectService.php | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/lib/Service/ObjectService.php b/lib/Service/ObjectService.php index bc626fb2..e0befbbf 100644 --- a/lib/Service/ObjectService.php +++ b/lib/Service/ObjectService.php @@ -12,6 +12,7 @@ use OCP\App\IAppManager; use OCP\AppFramework\Db\DoesNotExistException; use OCP\AppFramework\Db\MultipleObjectsReturnedException; +use OCP\IURLGenerator; use Psr\Container\ContainerExceptionInterface; use Psr\Container\NotFoundExceptionInterface; use Symfony\Component\Uid\Uuid; @@ -34,6 +35,8 @@ class ObjectService /** @var string $appName The name of the app */ private string $appName; + private ValidationService $validationService; + /** * Constructor for ObjectService. * @@ -59,8 +62,11 @@ public function __construct( private ContainerInterface $container, private readonly IAppManager $appManager, private readonly IAppConfig $config, + IURLGenerator $urlGenerator, ) { $this->appName = 'opencatalogi'; + + $this->validationService = new ValidationService(objectService: $this, urlGenerator: $urlGenerator); } /** From b91c66dfd0939955e846955116502237d2c5f5c8 Mon Sep 17 00:00:00 2001 From: Robert Zondervan Date: Fri, 8 Nov 2024 10:13:43 +0100 Subject: [PATCH 04/18] Fix routing error --- composer.json | 3 +- composer.lock | 868 +++++++++++++++++++++--------- lib/Db/PublicationType.php | 30 ++ lib/Service/ValidationService.php | 5 + 4 files changed, 645 insertions(+), 261 deletions(-) diff --git a/composer.json b/composer.json index 89c318a1..9672442e 100644 --- a/composer.json +++ b/composer.json @@ -44,8 +44,9 @@ "adbario/php-dot-notation": "^3.3.0", "bamarni/composer-bin-plugin": "^1.8", "elasticsearch/elasticsearch": "^v8.14.0", - "guzzlehttp/guzzle": "^7.0", + "guzzlehttp/guzzle": "^7.0", "mpdf/mpdf": "^8.2", + "opis/json-schema": "^2.3", "symfony/twig-bundle": "^6.4", "symfony/uid": "^6.4" }, diff --git a/composer.lock b/composer.lock index 7284d015..917ed994 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "93ceb09bbb9b85fb7d77d68d6d6eefe1", + "content-hash": "bd1fa2147c63c25a4c1759758bcf0fbb", "packages": [ { "name": "adbario/php-dot-notation", @@ -119,20 +119,21 @@ }, { "name": "elastic/transport", - "version": "v8.8.0", + "version": "v8.10.0", "source": { "type": "git", "url": "https://github.com/elastic/elastic-transport-php.git", - "reference": "cdf9f63a16ec6bfb4c881ab89aa0e2a61fb7c20b" + "reference": "8be37d679637545e50b1cea9f8ee903888783021" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/elastic/elastic-transport-php/zipball/cdf9f63a16ec6bfb4c881ab89aa0e2a61fb7c20b", - "reference": "cdf9f63a16ec6bfb4c881ab89aa0e2a61fb7c20b", + "url": "https://api.github.com/repos/elastic/elastic-transport-php/zipball/8be37d679637545e50b1cea9f8ee903888783021", + "reference": "8be37d679637545e50b1cea9f8ee903888783021", "shasum": "" }, "require": { "composer-runtime-api": "^2.0", + "open-telemetry/api": "^1.0", "php": "^7.4 || ^8.0", "php-http/discovery": "^1.14", "php-http/httplug": "^2.3", @@ -143,9 +144,11 @@ }, "require-dev": { "nyholm/psr7": "^1.5", + "open-telemetry/sdk": "^1.0", "php-http/mock-client": "^1.5", "phpstan/phpstan": "^1.4", - "phpunit/phpunit": "^9.5" + "phpunit/phpunit": "^9.5", + "symfony/http-client": "^5.4" }, "type": "library", "autoload": { @@ -168,26 +171,26 @@ ], "support": { "issues": "https://github.com/elastic/elastic-transport-php/issues", - "source": "https://github.com/elastic/elastic-transport-php/tree/v8.8.0" + "source": "https://github.com/elastic/elastic-transport-php/tree/v8.10.0" }, - "time": "2023-11-08T10:51:51+00:00" + "time": "2024-08-14T08:55:07+00:00" }, { "name": "elasticsearch/elasticsearch", - "version": "v8.14.0", + "version": "v8.15.0", "source": { "type": "git", "url": "https://github.com/elastic/elasticsearch-php.git", - "reference": "bff3c3e2402f6a20449404637f91a5ae214eff46" + "reference": "34c2444fa8d4c3e6c8b009bd8dea90bca007203b" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/elastic/elasticsearch-php/zipball/bff3c3e2402f6a20449404637f91a5ae214eff46", - "reference": "bff3c3e2402f6a20449404637f91a5ae214eff46", + "url": "https://api.github.com/repos/elastic/elasticsearch-php/zipball/34c2444fa8d4c3e6c8b009bd8dea90bca007203b", + "reference": "34c2444fa8d4c3e6c8b009bd8dea90bca007203b", "shasum": "" }, "require": { - "elastic/transport": "^8.8", + "elastic/transport": "^8.10", "guzzlehttp/guzzle": "^7.0", "php": "^7.4 || ^8.0", "psr/http-client": "^1.0", @@ -226,28 +229,28 @@ ], "support": { "issues": "https://github.com/elastic/elasticsearch-php/issues", - "source": "https://github.com/elastic/elasticsearch-php/tree/v8.14.0" + "source": "https://github.com/elastic/elasticsearch-php/tree/v8.15.0" }, - "time": "2024-06-12T19:58:31+00:00" + "time": "2024-08-14T14:32:50+00:00" }, { "name": "guzzlehttp/guzzle", - "version": "7.8.1", + "version": "7.9.2", "source": { "type": "git", "url": "https://github.com/guzzle/guzzle.git", - "reference": "41042bc7ab002487b876a0683fc8dce04ddce104" + "reference": "d281ed313b989f213357e3be1a179f02196ac99b" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/guzzle/guzzle/zipball/41042bc7ab002487b876a0683fc8dce04ddce104", - "reference": "41042bc7ab002487b876a0683fc8dce04ddce104", + "url": "https://api.github.com/repos/guzzle/guzzle/zipball/d281ed313b989f213357e3be1a179f02196ac99b", + "reference": "d281ed313b989f213357e3be1a179f02196ac99b", "shasum": "" }, "require": { "ext-json": "*", - "guzzlehttp/promises": "^1.5.3 || ^2.0.1", - "guzzlehttp/psr7": "^1.9.1 || ^2.5.1", + "guzzlehttp/promises": "^1.5.3 || ^2.0.3", + "guzzlehttp/psr7": "^2.7.0", "php": "^7.2.5 || ^8.0", "psr/http-client": "^1.0", "symfony/deprecation-contracts": "^2.2 || ^3.0" @@ -258,9 +261,9 @@ "require-dev": { "bamarni/composer-bin-plugin": "^1.8.2", "ext-curl": "*", - "php-http/client-integration-tests": "dev-master#2c025848417c1135031fdf9c728ee53d0a7ceaee as 3.0.999", + "guzzle/client-integration-tests": "3.0.2", "php-http/message-factory": "^1.1", - "phpunit/phpunit": "^8.5.36 || ^9.6.15", + "phpunit/phpunit": "^8.5.39 || ^9.6.20", "psr/log": "^1.1 || ^2.0 || ^3.0" }, "suggest": { @@ -338,7 +341,7 @@ ], "support": { "issues": "https://github.com/guzzle/guzzle/issues", - "source": "https://github.com/guzzle/guzzle/tree/7.8.1" + "source": "https://github.com/guzzle/guzzle/tree/7.9.2" }, "funding": [ { @@ -354,20 +357,20 @@ "type": "tidelift" } ], - "time": "2023-12-03T20:35:24+00:00" + "time": "2024-07-24T11:22:20+00:00" }, { "name": "guzzlehttp/promises", - "version": "2.0.2", + "version": "2.0.4", "source": { "type": "git", "url": "https://github.com/guzzle/promises.git", - "reference": "bbff78d96034045e58e13dedd6ad91b5d1253223" + "reference": "f9c436286ab2892c7db7be8c8da4ef61ccf7b455" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/guzzle/promises/zipball/bbff78d96034045e58e13dedd6ad91b5d1253223", - "reference": "bbff78d96034045e58e13dedd6ad91b5d1253223", + "url": "https://api.github.com/repos/guzzle/promises/zipball/f9c436286ab2892c7db7be8c8da4ef61ccf7b455", + "reference": "f9c436286ab2892c7db7be8c8da4ef61ccf7b455", "shasum": "" }, "require": { @@ -375,7 +378,7 @@ }, "require-dev": { "bamarni/composer-bin-plugin": "^1.8.2", - "phpunit/phpunit": "^8.5.36 || ^9.6.15" + "phpunit/phpunit": "^8.5.39 || ^9.6.20" }, "type": "library", "extra": { @@ -421,7 +424,7 @@ ], "support": { "issues": "https://github.com/guzzle/promises/issues", - "source": "https://github.com/guzzle/promises/tree/2.0.2" + "source": "https://github.com/guzzle/promises/tree/2.0.4" }, "funding": [ { @@ -437,20 +440,20 @@ "type": "tidelift" } ], - "time": "2023-12-03T20:19:20+00:00" + "time": "2024-10-17T10:06:22+00:00" }, { "name": "guzzlehttp/psr7", - "version": "2.6.2", + "version": "2.7.0", "source": { "type": "git", "url": "https://github.com/guzzle/psr7.git", - "reference": "45b30f99ac27b5ca93cb4831afe16285f57b8221" + "reference": "a70f5c95fb43bc83f07c9c948baa0dc1829bf201" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/guzzle/psr7/zipball/45b30f99ac27b5ca93cb4831afe16285f57b8221", - "reference": "45b30f99ac27b5ca93cb4831afe16285f57b8221", + "url": "https://api.github.com/repos/guzzle/psr7/zipball/a70f5c95fb43bc83f07c9c948baa0dc1829bf201", + "reference": "a70f5c95fb43bc83f07c9c948baa0dc1829bf201", "shasum": "" }, "require": { @@ -465,8 +468,8 @@ }, "require-dev": { "bamarni/composer-bin-plugin": "^1.8.2", - "http-interop/http-factory-tests": "^0.9", - "phpunit/phpunit": "^8.5.36 || ^9.6.15" + "http-interop/http-factory-tests": "0.9.0", + "phpunit/phpunit": "^8.5.39 || ^9.6.20" }, "suggest": { "laminas/laminas-httphandlerrunner": "Emit PSR-7 responses" @@ -537,7 +540,7 @@ ], "support": { "issues": "https://github.com/guzzle/psr7/issues", - "source": "https://github.com/guzzle/psr7/tree/2.6.2" + "source": "https://github.com/guzzle/psr7/tree/2.7.0" }, "funding": [ { @@ -553,7 +556,7 @@ "type": "tidelift" } ], - "time": "2023-12-03T20:05:35+00:00" + "time": "2024-07-18T11:15:46+00:00" }, { "name": "mpdf/mpdf", @@ -788,6 +791,325 @@ ], "time": "2024-06-12T14:39:25+00:00" }, + { + "name": "open-telemetry/api", + "version": "1.1.1", + "source": { + "type": "git", + "url": "https://github.com/opentelemetry-php/api.git", + "reference": "542064815d38a6df55af7957cd6f1d7d967c99c6" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/opentelemetry-php/api/zipball/542064815d38a6df55af7957cd6f1d7d967c99c6", + "reference": "542064815d38a6df55af7957cd6f1d7d967c99c6", + "shasum": "" + }, + "require": { + "open-telemetry/context": "^1.0", + "php": "^8.1", + "psr/log": "^1.1|^2.0|^3.0", + "symfony/polyfill-php82": "^1.26" + }, + "conflict": { + "open-telemetry/sdk": "<=1.0.8" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.1.x-dev" + }, + "spi": { + "OpenTelemetry\\API\\Instrumentation\\AutoInstrumentation\\HookManagerInterface": [ + "OpenTelemetry\\API\\Instrumentation\\AutoInstrumentation\\ExtensionHookManager" + ] + } + }, + "autoload": { + "files": [ + "Trace/functions.php" + ], + "psr-4": { + "OpenTelemetry\\API\\": "." + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "opentelemetry-php contributors", + "homepage": "https://github.com/open-telemetry/opentelemetry-php/graphs/contributors" + } + ], + "description": "API for OpenTelemetry PHP.", + "keywords": [ + "Metrics", + "api", + "apm", + "logging", + "opentelemetry", + "otel", + "tracing" + ], + "support": { + "chat": "https://app.slack.com/client/T08PSQ7BQ/C01NFPCV44V", + "docs": "https://opentelemetry.io/docs/php", + "issues": "https://github.com/open-telemetry/opentelemetry-php/issues", + "source": "https://github.com/open-telemetry/opentelemetry-php" + }, + "time": "2024-10-15T22:42:37+00:00" + }, + { + "name": "open-telemetry/context", + "version": "1.1.0", + "source": { + "type": "git", + "url": "https://github.com/opentelemetry-php/context.git", + "reference": "0cba875ea1953435f78aec7f1d75afa87bdbf7f3" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/opentelemetry-php/context/zipball/0cba875ea1953435f78aec7f1d75afa87bdbf7f3", + "reference": "0cba875ea1953435f78aec7f1d75afa87bdbf7f3", + "shasum": "" + }, + "require": { + "php": "^8.1", + "symfony/polyfill-php82": "^1.26" + }, + "suggest": { + "ext-ffi": "To allow context switching in Fibers" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.0.x-dev" + } + }, + "autoload": { + "files": [ + "fiber/initialize_fiber_handler.php" + ], + "psr-4": { + "OpenTelemetry\\Context\\": "." + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "opentelemetry-php contributors", + "homepage": "https://github.com/open-telemetry/opentelemetry-php/graphs/contributors" + } + ], + "description": "Context implementation for OpenTelemetry PHP.", + "keywords": [ + "Context", + "opentelemetry", + "otel" + ], + "support": { + "chat": "https://app.slack.com/client/T08PSQ7BQ/C01NFPCV44V", + "docs": "https://opentelemetry.io/docs/php", + "issues": "https://github.com/open-telemetry/opentelemetry-php/issues", + "source": "https://github.com/open-telemetry/opentelemetry-php" + }, + "time": "2024-08-21T00:29:20+00:00" + }, + { + "name": "opis/json-schema", + "version": "2.3.0", + "source": { + "type": "git", + "url": "https://github.com/opis/json-schema.git", + "reference": "c48df6d7089a45f01e1c82432348f2d5976f9bfb" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/opis/json-schema/zipball/c48df6d7089a45f01e1c82432348f2d5976f9bfb", + "reference": "c48df6d7089a45f01e1c82432348f2d5976f9bfb", + "shasum": "" + }, + "require": { + "ext-json": "*", + "opis/string": "^2.0", + "opis/uri": "^1.0", + "php": "^7.4 || ^8.0" + }, + "require-dev": { + "ext-bcmath": "*", + "ext-intl": "*", + "phpunit/phpunit": "^9.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.x-dev" + } + }, + "autoload": { + "psr-4": { + "Opis\\JsonSchema\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "Sorin Sarca", + "email": "sarca_sorin@hotmail.com" + }, + { + "name": "Marius Sarca", + "email": "marius.sarca@gmail.com" + } + ], + "description": "Json Schema Validator for PHP", + "homepage": "https://opis.io/json-schema", + "keywords": [ + "json", + "json-schema", + "schema", + "validation", + "validator" + ], + "support": { + "issues": "https://github.com/opis/json-schema/issues", + "source": "https://github.com/opis/json-schema/tree/2.3.0" + }, + "time": "2022-01-08T20:38:03+00:00" + }, + { + "name": "opis/string", + "version": "2.0.1", + "source": { + "type": "git", + "url": "https://github.com/opis/string.git", + "reference": "9ebf1a1f873f502f6859d11210b25a4bf5d141e7" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/opis/string/zipball/9ebf1a1f873f502f6859d11210b25a4bf5d141e7", + "reference": "9ebf1a1f873f502f6859d11210b25a4bf5d141e7", + "shasum": "" + }, + "require": { + "ext-iconv": "*", + "ext-json": "*", + "php": "^7.4 || ^8.0" + }, + "require-dev": { + "phpunit/phpunit": "^9.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.x-dev" + } + }, + "autoload": { + "psr-4": { + "Opis\\String\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "Marius Sarca", + "email": "marius.sarca@gmail.com" + }, + { + "name": "Sorin Sarca", + "email": "sarca_sorin@hotmail.com" + } + ], + "description": "Multibyte strings as objects", + "homepage": "https://opis.io/string", + "keywords": [ + "multi-byte", + "opis", + "string", + "string manipulation", + "utf-8" + ], + "support": { + "issues": "https://github.com/opis/string/issues", + "source": "https://github.com/opis/string/tree/2.0.1" + }, + "time": "2022-01-14T15:42:23+00:00" + }, + { + "name": "opis/uri", + "version": "1.1.0", + "source": { + "type": "git", + "url": "https://github.com/opis/uri.git", + "reference": "0f3ca49ab1a5e4a6681c286e0b2cc081b93a7d5a" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/opis/uri/zipball/0f3ca49ab1a5e4a6681c286e0b2cc081b93a7d5a", + "reference": "0f3ca49ab1a5e4a6681c286e0b2cc081b93a7d5a", + "shasum": "" + }, + "require": { + "opis/string": "^2.0", + "php": "^7.4 || ^8.0" + }, + "require-dev": { + "phpunit/phpunit": "^9" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.x-dev" + } + }, + "autoload": { + "psr-4": { + "Opis\\Uri\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "Marius Sarca", + "email": "marius.sarca@gmail.com" + }, + { + "name": "Sorin Sarca", + "email": "sarca_sorin@hotmail.com" + } + ], + "description": "Build, parse and validate URIs and URI-templates", + "homepage": "https://opis.io", + "keywords": [ + "URI Template", + "parse url", + "punycode", + "uri", + "uri components", + "url", + "validate uri" + ], + "support": { + "issues": "https://github.com/opis/uri/issues", + "source": "https://github.com/opis/uri/tree/1.1.0" + }, + "time": "2021-05-22T15:57:08+00:00" + }, { "name": "paragonie/random_compat", "version": "v9.99.100", @@ -840,16 +1162,16 @@ }, { "name": "php-http/discovery", - "version": "1.19.4", + "version": "1.20.0", "source": { "type": "git", "url": "https://github.com/php-http/discovery.git", - "reference": "0700efda8d7526335132360167315fdab3aeb599" + "reference": "82fe4c73ef3363caed49ff8dd1539ba06044910d" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/php-http/discovery/zipball/0700efda8d7526335132360167315fdab3aeb599", - "reference": "0700efda8d7526335132360167315fdab3aeb599", + "url": "https://api.github.com/repos/php-http/discovery/zipball/82fe4c73ef3363caed49ff8dd1539ba06044910d", + "reference": "82fe4c73ef3363caed49ff8dd1539ba06044910d", "shasum": "" }, "require": { @@ -913,22 +1235,22 @@ ], "support": { "issues": "https://github.com/php-http/discovery/issues", - "source": "https://github.com/php-http/discovery/tree/1.19.4" + "source": "https://github.com/php-http/discovery/tree/1.20.0" }, - "time": "2024-03-29T13:00:05+00:00" + "time": "2024-10-02T11:20:13+00:00" }, { "name": "php-http/httplug", - "version": "2.4.0", + "version": "2.4.1", "source": { "type": "git", "url": "https://github.com/php-http/httplug.git", - "reference": "625ad742c360c8ac580fcc647a1541d29e257f67" + "reference": "5cad731844891a4c282f3f3e1b582c46839d22f4" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/php-http/httplug/zipball/625ad742c360c8ac580fcc647a1541d29e257f67", - "reference": "625ad742c360c8ac580fcc647a1541d29e257f67", + "url": "https://api.github.com/repos/php-http/httplug/zipball/5cad731844891a4c282f3f3e1b582c46839d22f4", + "reference": "5cad731844891a4c282f3f3e1b582c46839d22f4", "shasum": "" }, "require": { @@ -970,9 +1292,9 @@ ], "support": { "issues": "https://github.com/php-http/httplug/issues", - "source": "https://github.com/php-http/httplug/tree/2.4.0" + "source": "https://github.com/php-http/httplug/tree/2.4.1" }, - "time": "2023-04-14T15:10:03+00:00" + "time": "2024-09-23T11:39:58+00:00" }, { "name": "php-http/promise", @@ -1385,16 +1707,16 @@ }, { "name": "setasign/fpdi", - "version": "v2.6.0", + "version": "v2.6.1", "source": { "type": "git", "url": "https://github.com/Setasign/FPDI.git", - "reference": "a6db878129ec6c7e141316ee71872923e7f1b7ad" + "reference": "09a816004fcee9ed3405bd164147e3fdbb79a56f" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/Setasign/FPDI/zipball/a6db878129ec6c7e141316ee71872923e7f1b7ad", - "reference": "a6db878129ec6c7e141316ee71872923e7f1b7ad", + "url": "https://api.github.com/repos/Setasign/FPDI/zipball/09a816004fcee9ed3405bd164147e3fdbb79a56f", + "reference": "09a816004fcee9ed3405bd164147e3fdbb79a56f", "shasum": "" }, "require": { @@ -1445,7 +1767,7 @@ ], "support": { "issues": "https://github.com/Setasign/FPDI/issues", - "source": "https://github.com/Setasign/FPDI/tree/v2.6.0" + "source": "https://github.com/Setasign/FPDI/tree/v2.6.1" }, "funding": [ { @@ -1453,20 +1775,20 @@ "type": "tidelift" } ], - "time": "2023-12-11T16:03:32+00:00" + "time": "2024-09-02T10:17:15+00:00" }, { "name": "symfony/config", - "version": "v6.4.8", + "version": "v6.4.14", "source": { "type": "git", "url": "https://github.com/symfony/config.git", - "reference": "12e7e52515ce37191b193cf3365903c4f3951e35" + "reference": "4e55e7e4ffddd343671ea972216d4509f46c22ef" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/config/zipball/12e7e52515ce37191b193cf3365903c4f3951e35", - "reference": "12e7e52515ce37191b193cf3365903c4f3951e35", + "url": "https://api.github.com/repos/symfony/config/zipball/4e55e7e4ffddd343671ea972216d4509f46c22ef", + "reference": "4e55e7e4ffddd343671ea972216d4509f46c22ef", "shasum": "" }, "require": { @@ -1512,7 +1834,7 @@ "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.4.8" + "source": "https://github.com/symfony/config/tree/v6.4.14" }, "funding": [ { @@ -1528,20 +1850,20 @@ "type": "tidelift" } ], - "time": "2024-05-31T14:49:08+00:00" + "time": "2024-11-04T11:33:53+00:00" }, { "name": "symfony/dependency-injection", - "version": "v6.4.10", + "version": "v6.4.13", "source": { "type": "git", "url": "https://github.com/symfony/dependency-injection.git", - "reference": "5caf9c5f6085f13b27d70a236b776c07e4a1c3eb" + "reference": "728ae8f4e190133ce99d6d5f0bc1e8c8bd7c7a96" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/dependency-injection/zipball/5caf9c5f6085f13b27d70a236b776c07e4a1c3eb", - "reference": "5caf9c5f6085f13b27d70a236b776c07e4a1c3eb", + "url": "https://api.github.com/repos/symfony/dependency-injection/zipball/728ae8f4e190133ce99d6d5f0bc1e8c8bd7c7a96", + "reference": "728ae8f4e190133ce99d6d5f0bc1e8c8bd7c7a96", "shasum": "" }, "require": { @@ -1593,7 +1915,7 @@ "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.4.10" + "source": "https://github.com/symfony/dependency-injection/tree/v6.4.13" }, "funding": [ { @@ -1609,7 +1931,7 @@ "type": "tidelift" } ], - "time": "2024-07-26T07:32:07+00:00" + "time": "2024-10-25T15:07:50+00:00" }, { "name": "symfony/deprecation-contracts", @@ -1680,16 +2002,16 @@ }, { "name": "symfony/error-handler", - "version": "v6.4.10", + "version": "v6.4.14", "source": { "type": "git", "url": "https://github.com/symfony/error-handler.git", - "reference": "231f1b2ee80f72daa1972f7340297d67439224f0" + "reference": "9e024324511eeb00983ee76b9aedc3e6ecd993d9" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/error-handler/zipball/231f1b2ee80f72daa1972f7340297d67439224f0", - "reference": "231f1b2ee80f72daa1972f7340297d67439224f0", + "url": "https://api.github.com/repos/symfony/error-handler/zipball/9e024324511eeb00983ee76b9aedc3e6ecd993d9", + "reference": "9e024324511eeb00983ee76b9aedc3e6ecd993d9", "shasum": "" }, "require": { @@ -1735,7 +2057,7 @@ "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.4.10" + "source": "https://github.com/symfony/error-handler/tree/v6.4.14" }, "funding": [ { @@ -1751,20 +2073,20 @@ "type": "tidelift" } ], - "time": "2024-07-26T12:30:32+00:00" + "time": "2024-11-05T15:34:40+00:00" }, { "name": "symfony/event-dispatcher", - "version": "v6.4.8", + "version": "v6.4.13", "source": { "type": "git", "url": "https://github.com/symfony/event-dispatcher.git", - "reference": "8d7507f02b06e06815e56bb39aa0128e3806208b" + "reference": "0ffc48080ab3e9132ea74ef4e09d8dcf26bf897e" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/8d7507f02b06e06815e56bb39aa0128e3806208b", - "reference": "8d7507f02b06e06815e56bb39aa0128e3806208b", + "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/0ffc48080ab3e9132ea74ef4e09d8dcf26bf897e", + "reference": "0ffc48080ab3e9132ea74ef4e09d8dcf26bf897e", "shasum": "" }, "require": { @@ -1815,7 +2137,7 @@ "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.4.8" + "source": "https://github.com/symfony/event-dispatcher/tree/v6.4.13" }, "funding": [ { @@ -1831,7 +2153,7 @@ "type": "tidelift" } ], - "time": "2024-05-31T14:49:08+00:00" + "time": "2024-09-25T14:18:03+00:00" }, { "name": "symfony/event-dispatcher-contracts", @@ -1911,16 +2233,16 @@ }, { "name": "symfony/filesystem", - "version": "v6.4.9", + "version": "v6.4.13", "source": { "type": "git", "url": "https://github.com/symfony/filesystem.git", - "reference": "b51ef8059159330b74a4d52f68e671033c0fe463" + "reference": "4856c9cf585d5a0313d8d35afd681a526f038dd3" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/filesystem/zipball/b51ef8059159330b74a4d52f68e671033c0fe463", - "reference": "b51ef8059159330b74a4d52f68e671033c0fe463", + "url": "https://api.github.com/repos/symfony/filesystem/zipball/4856c9cf585d5a0313d8d35afd681a526f038dd3", + "reference": "4856c9cf585d5a0313d8d35afd681a526f038dd3", "shasum": "" }, "require": { @@ -1957,7 +2279,7 @@ "description": "Provides basic utilities for the filesystem", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/filesystem/tree/v6.4.9" + "source": "https://github.com/symfony/filesystem/tree/v6.4.13" }, "funding": [ { @@ -1973,20 +2295,20 @@ "type": "tidelift" } ], - "time": "2024-06-28T09:49:33+00:00" + "time": "2024-10-25T15:07:50+00:00" }, { "name": "symfony/http-foundation", - "version": "v6.4.10", + "version": "v6.4.14", "source": { "type": "git", "url": "https://github.com/symfony/http-foundation.git", - "reference": "117f1f20a7ade7bcea28b861fb79160a21a1e37b" + "reference": "ba020a321a95519303a3f09ec2824d34d601c388" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/http-foundation/zipball/117f1f20a7ade7bcea28b861fb79160a21a1e37b", - "reference": "117f1f20a7ade7bcea28b861fb79160a21a1e37b", + "url": "https://api.github.com/repos/symfony/http-foundation/zipball/ba020a321a95519303a3f09ec2824d34d601c388", + "reference": "ba020a321a95519303a3f09ec2824d34d601c388", "shasum": "" }, "require": { @@ -2034,7 +2356,7 @@ "description": "Defines an object-oriented layer for the HTTP specification", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/http-foundation/tree/v6.4.10" + "source": "https://github.com/symfony/http-foundation/tree/v6.4.14" }, "funding": [ { @@ -2050,20 +2372,20 @@ "type": "tidelift" } ], - "time": "2024-07-26T12:36:27+00:00" + "time": "2024-11-05T16:39:55+00:00" }, { "name": "symfony/http-kernel", - "version": "v6.4.10", + "version": "v6.4.14", "source": { "type": "git", "url": "https://github.com/symfony/http-kernel.git", - "reference": "147e0daf618d7575b5007055340d09aece5cf068" + "reference": "8278a947d0369754a47b758a9e17b72cab970951" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/http-kernel/zipball/147e0daf618d7575b5007055340d09aece5cf068", - "reference": "147e0daf618d7575b5007055340d09aece5cf068", + "url": "https://api.github.com/repos/symfony/http-kernel/zipball/8278a947d0369754a47b758a9e17b72cab970951", + "reference": "8278a947d0369754a47b758a9e17b72cab970951", "shasum": "" }, "require": { @@ -2148,7 +2470,7 @@ "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.4.10" + "source": "https://github.com/symfony/http-kernel/tree/v6.4.14" }, "funding": [ { @@ -2164,24 +2486,24 @@ "type": "tidelift" } ], - "time": "2024-07-26T14:52:04+00:00" + "time": "2024-11-06T09:45:21+00:00" }, { "name": "symfony/polyfill-ctype", - "version": "v1.30.0", + "version": "v1.31.0", "source": { "type": "git", "url": "https://github.com/symfony/polyfill-ctype.git", - "reference": "0424dff1c58f028c451efff2045f5d92410bd540" + "reference": "a3cc8b044a6ea513310cbd48ef7333b384945638" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-ctype/zipball/0424dff1c58f028c451efff2045f5d92410bd540", - "reference": "0424dff1c58f028c451efff2045f5d92410bd540", + "url": "https://api.github.com/repos/symfony/polyfill-ctype/zipball/a3cc8b044a6ea513310cbd48ef7333b384945638", + "reference": "a3cc8b044a6ea513310cbd48ef7333b384945638", "shasum": "" }, "require": { - "php": ">=7.1" + "php": ">=7.2" }, "provide": { "ext-ctype": "*" @@ -2227,7 +2549,7 @@ "portable" ], "support": { - "source": "https://github.com/symfony/polyfill-ctype/tree/v1.30.0" + "source": "https://github.com/symfony/polyfill-ctype/tree/v1.31.0" }, "funding": [ { @@ -2243,24 +2565,24 @@ "type": "tidelift" } ], - "time": "2024-05-31T15:07:36+00:00" + "time": "2024-09-09T11:45:10+00:00" }, { "name": "symfony/polyfill-mbstring", - "version": "v1.30.0", + "version": "v1.31.0", "source": { "type": "git", "url": "https://github.com/symfony/polyfill-mbstring.git", - "reference": "fd22ab50000ef01661e2a31d850ebaa297f8e03c" + "reference": "85181ba99b2345b0ef10ce42ecac37612d9fd341" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-mbstring/zipball/fd22ab50000ef01661e2a31d850ebaa297f8e03c", - "reference": "fd22ab50000ef01661e2a31d850ebaa297f8e03c", + "url": "https://api.github.com/repos/symfony/polyfill-mbstring/zipball/85181ba99b2345b0ef10ce42ecac37612d9fd341", + "reference": "85181ba99b2345b0ef10ce42ecac37612d9fd341", "shasum": "" }, "require": { - "php": ">=7.1" + "php": ">=7.2" }, "provide": { "ext-mbstring": "*" @@ -2307,7 +2629,7 @@ "shim" ], "support": { - "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.30.0" + "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.31.0" }, "funding": [ { @@ -2323,24 +2645,24 @@ "type": "tidelift" } ], - "time": "2024-06-19T12:30:46+00:00" + "time": "2024-09-09T11:45:10+00:00" }, { - "name": "symfony/polyfill-php80", - "version": "v1.30.0", + "name": "symfony/polyfill-php81", + "version": "v1.31.0", "source": { "type": "git", - "url": "https://github.com/symfony/polyfill-php80.git", - "reference": "77fa7995ac1b21ab60769b7323d600a991a90433" + "url": "https://github.com/symfony/polyfill-php81.git", + "reference": "4a4cfc2d253c21a5ad0e53071df248ed48c6ce5c" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-php80/zipball/77fa7995ac1b21ab60769b7323d600a991a90433", - "reference": "77fa7995ac1b21ab60769b7323d600a991a90433", + "url": "https://api.github.com/repos/symfony/polyfill-php81/zipball/4a4cfc2d253c21a5ad0e53071df248ed48c6ce5c", + "reference": "4a4cfc2d253c21a5ad0e53071df248ed48c6ce5c", "shasum": "" }, "require": { - "php": ">=7.1" + "php": ">=7.2" }, "type": "library", "extra": { @@ -2354,7 +2676,7 @@ "bootstrap.php" ], "psr-4": { - "Symfony\\Polyfill\\Php80\\": "" + "Symfony\\Polyfill\\Php81\\": "" }, "classmap": [ "Resources/stubs" @@ -2365,10 +2687,6 @@ "MIT" ], "authors": [ - { - "name": "Ion Bazan", - "email": "ion.bazan@gmail.com" - }, { "name": "Nicolas Grekas", "email": "p@tchwork.com" @@ -2378,7 +2696,7 @@ "homepage": "https://symfony.com/contributors" } ], - "description": "Symfony polyfill backporting some PHP 8.0+ features to lower PHP versions", + "description": "Symfony polyfill backporting some PHP 8.1+ features to lower PHP versions", "homepage": "https://symfony.com", "keywords": [ "compatibility", @@ -2387,7 +2705,7 @@ "shim" ], "support": { - "source": "https://github.com/symfony/polyfill-php80/tree/v1.30.0" + "source": "https://github.com/symfony/polyfill-php81/tree/v1.31.0" }, "funding": [ { @@ -2403,24 +2721,24 @@ "type": "tidelift" } ], - "time": "2024-05-31T15:07:36+00:00" + "time": "2024-09-09T11:45:10+00:00" }, { - "name": "symfony/polyfill-php81", - "version": "v1.30.0", + "name": "symfony/polyfill-php82", + "version": "v1.31.0", "source": { "type": "git", - "url": "https://github.com/symfony/polyfill-php81.git", - "reference": "3fb075789fb91f9ad9af537c4012d523085bd5af" + "url": "https://github.com/symfony/polyfill-php82.git", + "reference": "5d2ed36f7734637dacc025f179698031951b1692" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-php81/zipball/3fb075789fb91f9ad9af537c4012d523085bd5af", - "reference": "3fb075789fb91f9ad9af537c4012d523085bd5af", + "url": "https://api.github.com/repos/symfony/polyfill-php82/zipball/5d2ed36f7734637dacc025f179698031951b1692", + "reference": "5d2ed36f7734637dacc025f179698031951b1692", "shasum": "" }, "require": { - "php": ">=7.1" + "php": ">=7.2" }, "type": "library", "extra": { @@ -2434,7 +2752,7 @@ "bootstrap.php" ], "psr-4": { - "Symfony\\Polyfill\\Php81\\": "" + "Symfony\\Polyfill\\Php82\\": "" }, "classmap": [ "Resources/stubs" @@ -2454,7 +2772,7 @@ "homepage": "https://symfony.com/contributors" } ], - "description": "Symfony polyfill backporting some PHP 8.1+ features to lower PHP versions", + "description": "Symfony polyfill backporting some PHP 8.2+ features to lower PHP versions", "homepage": "https://symfony.com", "keywords": [ "compatibility", @@ -2463,7 +2781,7 @@ "shim" ], "support": { - "source": "https://github.com/symfony/polyfill-php81/tree/v1.30.0" + "source": "https://github.com/symfony/polyfill-php82/tree/v1.31.0" }, "funding": [ { @@ -2479,24 +2797,24 @@ "type": "tidelift" } ], - "time": "2024-06-19T12:30:46+00:00" + "time": "2024-09-09T11:45:10+00:00" }, { "name": "symfony/polyfill-php83", - "version": "v1.30.0", + "version": "v1.31.0", "source": { "type": "git", "url": "https://github.com/symfony/polyfill-php83.git", - "reference": "dbdcdf1a4dcc2743591f1079d0c35ab1e2dcbbc9" + "reference": "2fb86d65e2d424369ad2905e83b236a8805ba491" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-php83/zipball/dbdcdf1a4dcc2743591f1079d0c35ab1e2dcbbc9", - "reference": "dbdcdf1a4dcc2743591f1079d0c35ab1e2dcbbc9", + "url": "https://api.github.com/repos/symfony/polyfill-php83/zipball/2fb86d65e2d424369ad2905e83b236a8805ba491", + "reference": "2fb86d65e2d424369ad2905e83b236a8805ba491", "shasum": "" }, "require": { - "php": ">=7.1" + "php": ">=7.2" }, "type": "library", "extra": { @@ -2539,7 +2857,7 @@ "shim" ], "support": { - "source": "https://github.com/symfony/polyfill-php83/tree/v1.30.0" + "source": "https://github.com/symfony/polyfill-php83/tree/v1.31.0" }, "funding": [ { @@ -2555,24 +2873,24 @@ "type": "tidelift" } ], - "time": "2024-06-19T12:35:24+00:00" + "time": "2024-09-09T11:45:10+00:00" }, { "name": "symfony/polyfill-uuid", - "version": "v1.30.0", + "version": "v1.31.0", "source": { "type": "git", "url": "https://github.com/symfony/polyfill-uuid.git", - "reference": "2ba1f33797470debcda07fe9dce20a0003df18e9" + "reference": "21533be36c24be3f4b1669c4725c7d1d2bab4ae2" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-uuid/zipball/2ba1f33797470debcda07fe9dce20a0003df18e9", - "reference": "2ba1f33797470debcda07fe9dce20a0003df18e9", + "url": "https://api.github.com/repos/symfony/polyfill-uuid/zipball/21533be36c24be3f4b1669c4725c7d1d2bab4ae2", + "reference": "21533be36c24be3f4b1669c4725c7d1d2bab4ae2", "shasum": "" }, "require": { - "php": ">=7.1" + "php": ">=7.2" }, "provide": { "ext-uuid": "*" @@ -2618,7 +2936,7 @@ "uuid" ], "support": { - "source": "https://github.com/symfony/polyfill-uuid/tree/v1.30.0" + "source": "https://github.com/symfony/polyfill-uuid/tree/v1.31.0" }, "funding": [ { @@ -2634,7 +2952,7 @@ "type": "tidelift" } ], - "time": "2024-05-31T15:07:36+00:00" + "time": "2024-09-09T11:45:10+00:00" }, { "name": "symfony/service-contracts", @@ -2799,16 +3117,16 @@ }, { "name": "symfony/twig-bridge", - "version": "v6.4.9", + "version": "v6.4.13", "source": { "type": "git", "url": "https://github.com/symfony/twig-bridge.git", - "reference": "9bcb26445b9d4ef1087c389234bf33fb00e10ea6" + "reference": "ec3511eef0576f378b2758da9e1c157086babd59" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/twig-bridge/zipball/9bcb26445b9d4ef1087c389234bf33fb00e10ea6", - "reference": "9bcb26445b9d4ef1087c389234bf33fb00e10ea6", + "url": "https://api.github.com/repos/symfony/twig-bridge/zipball/ec3511eef0576f378b2758da9e1c157086babd59", + "reference": "ec3511eef0576f378b2758da9e1c157086babd59", "shasum": "" }, "require": { @@ -2888,7 +3206,7 @@ "description": "Provides integration for Twig with various Symfony components", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/twig-bridge/tree/v6.4.9" + "source": "https://github.com/symfony/twig-bridge/tree/v6.4.13" }, "funding": [ { @@ -2904,20 +3222,20 @@ "type": "tidelift" } ], - "time": "2024-06-21T16:04:15+00:00" + "time": "2024-10-25T15:07:50+00:00" }, { "name": "symfony/twig-bundle", - "version": "v6.4.8", + "version": "v6.4.13", "source": { "type": "git", "url": "https://github.com/symfony/twig-bundle.git", - "reference": "ef17bc8fc2cb2376b235cd1b98f0275a78c5ba65" + "reference": "c3beeb5336aba1ea03c37e526968c2fde3ef25c4" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/twig-bundle/zipball/ef17bc8fc2cb2376b235cd1b98f0275a78c5ba65", - "reference": "ef17bc8fc2cb2376b235cd1b98f0275a78c5ba65", + "url": "https://api.github.com/repos/symfony/twig-bundle/zipball/c3beeb5336aba1ea03c37e526968c2fde3ef25c4", + "reference": "c3beeb5336aba1ea03c37e526968c2fde3ef25c4", "shasum": "" }, "require": { @@ -2972,7 +3290,7 @@ "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.4.8" + "source": "https://github.com/symfony/twig-bundle/tree/v6.4.13" }, "funding": [ { @@ -2988,20 +3306,20 @@ "type": "tidelift" } ], - "time": "2024-05-31T14:49:08+00:00" + "time": "2024-09-25T14:18:03+00:00" }, { "name": "symfony/uid", - "version": "v6.4.8", + "version": "v6.4.13", "source": { "type": "git", "url": "https://github.com/symfony/uid.git", - "reference": "35904eca37a84bb764c560cbfcac9f0ac2bcdbdf" + "reference": "18eb207f0436a993fffbdd811b5b8fa35fa5e007" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/uid/zipball/35904eca37a84bb764c560cbfcac9f0ac2bcdbdf", - "reference": "35904eca37a84bb764c560cbfcac9f0ac2bcdbdf", + "url": "https://api.github.com/repos/symfony/uid/zipball/18eb207f0436a993fffbdd811b5b8fa35fa5e007", + "reference": "18eb207f0436a993fffbdd811b5b8fa35fa5e007", "shasum": "" }, "require": { @@ -3046,7 +3364,7 @@ "uuid" ], "support": { - "source": "https://github.com/symfony/uid/tree/v6.4.8" + "source": "https://github.com/symfony/uid/tree/v6.4.13" }, "funding": [ { @@ -3062,20 +3380,20 @@ "type": "tidelift" } ], - "time": "2024-05-31T14:49:08+00:00" + "time": "2024-09-25T14:18:03+00:00" }, { "name": "symfony/var-dumper", - "version": "v6.4.10", + "version": "v6.4.14", "source": { "type": "git", "url": "https://github.com/symfony/var-dumper.git", - "reference": "a71cc3374f5fb9759da1961d28c452373b343dd4" + "reference": "93c09246038178717a9c14b809ea8151ffcf7091" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/var-dumper/zipball/a71cc3374f5fb9759da1961d28c452373b343dd4", - "reference": "a71cc3374f5fb9759da1961d28c452373b343dd4", + "url": "https://api.github.com/repos/symfony/var-dumper/zipball/93c09246038178717a9c14b809ea8151ffcf7091", + "reference": "93c09246038178717a9c14b809ea8151ffcf7091", "shasum": "" }, "require": { @@ -3131,7 +3449,7 @@ "dump" ], "support": { - "source": "https://github.com/symfony/var-dumper/tree/v6.4.10" + "source": "https://github.com/symfony/var-dumper/tree/v6.4.14" }, "funding": [ { @@ -3147,20 +3465,20 @@ "type": "tidelift" } ], - "time": "2024-07-26T12:30:32+00:00" + "time": "2024-11-05T15:34:40+00:00" }, { "name": "symfony/var-exporter", - "version": "v6.4.9", + "version": "v6.4.13", "source": { "type": "git", "url": "https://github.com/symfony/var-exporter.git", - "reference": "f9a060622e0d93777b7f8687ec4860191e16802e" + "reference": "0f605f72a363f8743001038a176eeb2a11223b51" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/var-exporter/zipball/f9a060622e0d93777b7f8687ec4860191e16802e", - "reference": "f9a060622e0d93777b7f8687ec4860191e16802e", + "url": "https://api.github.com/repos/symfony/var-exporter/zipball/0f605f72a363f8743001038a176eeb2a11223b51", + "reference": "0f605f72a363f8743001038a176eeb2a11223b51", "shasum": "" }, "require": { @@ -3208,7 +3526,7 @@ "serialize" ], "support": { - "source": "https://github.com/symfony/var-exporter/tree/v6.4.9" + "source": "https://github.com/symfony/var-exporter/tree/v6.4.13" }, "funding": [ { @@ -3224,28 +3542,27 @@ "type": "tidelift" } ], - "time": "2024-06-24T15:53:56+00:00" + "time": "2024-09-25T14:18:03+00:00" }, { "name": "twig/twig", - "version": "v3.11.0", + "version": "v3.14.2", "source": { "type": "git", "url": "https://github.com/twigphp/Twig.git", - "reference": "e80fb8ebba85c7341a97a9ebf825d7fd4b77708d" + "reference": "0b6f9d8370bb3b7f1ce5313ed8feb0fafd6e399a" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/twigphp/Twig/zipball/e80fb8ebba85c7341a97a9ebf825d7fd4b77708d", - "reference": "e80fb8ebba85c7341a97a9ebf825d7fd4b77708d", + "url": "https://api.github.com/repos/twigphp/Twig/zipball/0b6f9d8370bb3b7f1ce5313ed8feb0fafd6e399a", + "reference": "0b6f9d8370bb3b7f1ce5313ed8feb0fafd6e399a", "shasum": "" }, "require": { - "php": ">=7.2.5", + "php": ">=8.0.2", "symfony/deprecation-contracts": "^2.5|^3", "symfony/polyfill-ctype": "^1.8", "symfony/polyfill-mbstring": "^1.3", - "symfony/polyfill-php80": "^1.22", "symfony/polyfill-php81": "^1.29" }, "require-dev": { @@ -3292,7 +3609,7 @@ ], "support": { "issues": "https://github.com/twigphp/Twig/issues", - "source": "https://github.com/twigphp/Twig/tree/v3.11.0" + "source": "https://github.com/twigphp/Twig/tree/v3.14.2" }, "funding": [ { @@ -3304,7 +3621,7 @@ "type": "tidelift" } ], - "time": "2024-08-08T16:15:16+00:00" + "time": "2024-11-07T12:36:22+00:00" } ], "packages-dev": [ @@ -3314,12 +3631,12 @@ "source": { "type": "git", "url": "https://github.com/nextcloud-deps/ocp.git", - "reference": "65b6744fca5d4b3c366754295e5cb0680a580c51" + "reference": "5054ce1e0018c7f0946df391a54861f8172d7be2" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/nextcloud-deps/ocp/zipball/65b6744fca5d4b3c366754295e5cb0680a580c51", - "reference": "65b6744fca5d4b3c366754295e5cb0680a580c51", + "url": "https://api.github.com/repos/nextcloud-deps/ocp/zipball/5054ce1e0018c7f0946df391a54861f8172d7be2", + "reference": "5054ce1e0018c7f0946df391a54861f8172d7be2", "shasum": "" }, "require": { @@ -3350,7 +3667,7 @@ "issues": "https://github.com/nextcloud-deps/ocp/issues", "source": "https://github.com/nextcloud-deps/ocp/tree/stable29" }, - "time": "2024-07-11T00:37:34+00:00" + "time": "2024-09-17T00:34:06+00:00" }, { "name": "psr/clock", @@ -3406,23 +3723,23 @@ "source": { "type": "git", "url": "https://github.com/Roave/SecurityAdvisories.git", - "reference": "d44cb505c9515db5ff905ec510d69a0f80b366a2" + "reference": "e63317470a1b96346be224a68f9e64567e1001c3" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/Roave/SecurityAdvisories/zipball/d44cb505c9515db5ff905ec510d69a0f80b366a2", - "reference": "d44cb505c9515db5ff905ec510d69a0f80b366a2", + "url": "https://api.github.com/repos/Roave/SecurityAdvisories/zipball/e63317470a1b96346be224a68f9e64567e1001c3", + "reference": "e63317470a1b96346be224a68f9e64567e1001c3", "shasum": "" }, "conflict": { "3f/pygmentize": "<1.2", - "admidio/admidio": "<4.2.13", + "admidio/admidio": "<4.3.12", "adodb/adodb-php": "<=5.20.20|>=5.21,<=5.21.3", "aheinze/cockpit": "<2.2", - "aimeos/ai-admin-graphql": ">=2022.04.1,<2022.10.10|>=2023.04.1,<2023.10.6|>=2024.04.1,<2024.04.6", + "aimeos/ai-admin-graphql": ">=2022.04.1,<2022.10.10|>=2023.04.1,<2023.10.6|>=2024.04.1,<2024.07.2", "aimeos/ai-admin-jsonadm": "<2020.10.13|>=2021.04.1,<2021.10.6|>=2022.04.1,<2022.10.3|>=2023.04.1,<2023.10.4|==2024.04.1", "aimeos/ai-client-html": ">=2020.04.1,<2020.10.27|>=2021.04.1,<2021.10.22|>=2022.04.1,<2022.10.13|>=2023.04.1,<2023.10.15|>=2024.04.1,<2024.04.7", - "aimeos/ai-controller-frontend": "<2020.10.15|>=2021.04.1,<2021.10.8|>=2022.04.1,<2022.10.8|>=2023.04.1,<2023.10.9", + "aimeos/ai-controller-frontend": "<2020.10.15|>=2021.04.1,<2021.10.8|>=2022.04.1,<2022.10.8|>=2023.04.1,<2023.10.9|==2024.04.1", "aimeos/aimeos-core": ">=2022.04.1,<2022.10.17|>=2023.04.1,<2023.10.17|>=2024.04.1,<2024.04.7", "aimeos/aimeos-typo3": "<19.10.12|>=20,<20.10.5", "airesvsg/acf-to-rest-api": "<=3.1", @@ -3431,6 +3748,7 @@ "alextselegidis/easyappointments": "<1.5", "alterphp/easyadmin-extension-bundle": ">=1.2,<1.2.11|>=1.3,<1.3.1", "amazing/media2click": ">=1,<1.3.3", + "ameos/ameos_tarteaucitron": "<1.2.23", "amphp/artax": "<1.0.6|>=2,<2.0.6", "amphp/http": "<=1.7.2|>=2,<=2.1", "amphp/http-client": ">=4,<4.4", @@ -3449,12 +3767,12 @@ "athlon1600/php-proxy-app": "<=3", "austintoddj/canvas": "<=3.4.2", "auth0/wordpress": "<=4.6", - "automad/automad": "<=1.10.9", + "automad/automad": "<2.0.0.0-alpha5", "automattic/jetpack": "<9.8", "awesome-support/awesome-support": "<=6.0.7", "aws/aws-sdk-php": "<3.288.1", "azuracast/azuracast": "<0.18.3", - "backdrop/backdrop": "<1.24.2", + "backdrop/backdrop": "<1.27.3|>=1.28,<1.28.2", "backpack/crud": "<3.4.9", "bacula-web/bacula-web": "<8.0.0.0-RC2-dev", "badaso/core": "<2.7", @@ -3463,13 +3781,13 @@ "barrelstrength/sprout-forms": "<3.9", "barryvdh/laravel-translation-manager": "<0.6.2", "barzahlen/barzahlen-php": "<2.0.1", - "baserproject/basercms": "<5.0.9", + "baserproject/basercms": "<=5.1.1", "bassjobsen/bootstrap-3-typeahead": ">4.0.2", "bbpress/bbpress": "<2.6.5", "bcosca/fatfree": "<3.7.2", "bedita/bedita": "<4", "bigfork/silverstripe-form-capture": ">=3,<3.1.1", - "billz/raspap-webgui": "<2.9.5", + "billz/raspap-webgui": "<=3.1.4", "bk2k/bootstrap-package": ">=7.1,<7.1.2|>=8,<8.0.8|>=9,<9.0.4|>=9.1,<9.1.3|>=10,<10.0.10|>=11,<11.0.3", "blueimp/jquery-file-upload": "==6.4.4", "bmarshall511/wordpress_zero_spam": "<5.2.13", @@ -3508,31 +3826,34 @@ "codeigniter4/shield": "<1.0.0.0-beta8", "codiad/codiad": "<=2.8.4", "composer/composer": "<1.10.27|>=2,<2.2.24|>=2.3,<2.7.7", - "concrete5/concrete5": "<9.2.8", + "concrete5/concrete5": "<9.3.4", "concrete5/core": "<8.5.8|>=9,<9.1", "contao-components/mediaelement": ">=2.14.2,<2.21.1", "contao/comments-bundle": ">=2,<4.13.40|>=5.0.0.0-RC1-dev,<5.3.4", - "contao/contao": ">=3,<3.5.37|>=4,<4.4.56|>=4.5,<4.9.40|>=4.10,<4.11.7|>=4.13,<4.13.21|>=5.1,<5.1.4", + "contao/contao": "<=5.4.1", "contao/core": "<3.5.39", - "contao/core-bundle": "<4.13.40|>=5,<5.3.4", + "contao/core-bundle": "<4.13.49|>=5,<5.3.15|>=5.4,<5.4.3", "contao/listing-bundle": ">=3,<=3.5.30|>=4,<4.4.8", "contao/managed-edition": "<=1.5", "corveda/phpsandbox": "<1.3.5", "cosenary/instagram": "<=2.3", - "craftcms/cms": "<4.6.2", + "craftcms/cms": "<4.6.2|>=5,<=5.2.2", "croogo/croogo": "<4", "cuyz/valinor": "<0.12", + "czim/file-handling": "<1.5|>=2,<2.3", "czproject/git-php": "<4.0.3", + "damienharper/auditor-bundle": "<5.2.6", "dapphp/securimage": "<3.6.6", "darylldoyle/safe-svg": "<1.9.10", "datadog/dd-trace": ">=0.30,<0.30.2", "datatables/datatables": "<1.10.10", "david-garcia/phpwhois": "<=4.3.1", "dbrisinajumi/d2files": "<1", - "dcat/laravel-admin": "<=2.1.3.0-beta", + "dcat/laravel-admin": "<=2.1.3", "derhansen/fe_change_pwd": "<2.0.5|>=3,<3.0.3", "derhansen/sf_event_mgt": "<4.3.1|>=5,<5.1.1|>=7,<7.4", "desperado/xml-bundle": "<=0.1.7", + "dev-lancer/minecraft-motd-parser": "<=1.0.5", "devgroup/dotplant": "<2020.09.14-dev", "directmailteam/direct-mail": "<6.0.3|>=7,<7.0.3|>=8,<9.5.2", "doctrine/annotations": "<1.2.7", @@ -3547,8 +3868,9 @@ "dolibarr/dolibarr": "<19.0.2", "dompdf/dompdf": "<2.0.4", "doublethreedigital/guest-entries": "<3.1.2", - "drupal/core": ">=6,<6.38|>=7,<7.96|>=8,<10.1.8|>=10.2,<10.2.2", - "drupal/drupal": ">=5,<5.11|>=6,<6.38|>=7,<7.80|>=8,<8.9.16|>=9,<9.1.12|>=9.2,<9.2.4", + "drupal/core": ">=6,<6.38|>=7,<7.96|>=8,<10.2.9|>=10.3,<10.3.6|>=11,<11.0.5", + "drupal/core-recommended": ">=8,<10.2.9|>=10.3,<10.3.6|>=11,<11.0.5", + "drupal/drupal": ">=5,<5.11|>=6,<6.38|>=7,<7.80|>=8,<10.2.9|>=10.3,<10.3.6|>=11,<11.0.5", "duncanmcclean/guest-entries": "<3.1.2", "dweeves/magmi": "<=0.7.24", "ec-cube/ec-cube": "<2.4.4|>=2.11,<=2.17.1|>=3,<=3.0.18.0-patch4|>=4,<=4.1.2", @@ -3572,29 +3894,33 @@ "ezsystems/ezdemo-ls-extension": ">=5.4,<5.4.2.1-dev", "ezsystems/ezfind-ls": ">=5.3,<5.3.6.1-dev|>=5.4,<5.4.11.1-dev|>=2017.12,<2017.12.0.1-dev", "ezsystems/ezplatform": "<=1.13.6|>=2,<=2.5.24", - "ezsystems/ezplatform-admin-ui": ">=1.3,<1.3.5|>=1.4,<1.4.6|>=1.5,<1.5.29|>=2.3,<2.3.26", + "ezsystems/ezplatform-admin-ui": ">=1.3,<1.3.5|>=1.4,<1.4.6|>=1.5,<1.5.29|>=2.3,<2.3.26|>=3.3,<3.3.39", "ezsystems/ezplatform-admin-ui-assets": ">=4,<4.2.1|>=5,<5.0.1|>=5.1,<5.1.1", "ezsystems/ezplatform-graphql": ">=1.0.0.0-RC1-dev,<1.0.13|>=2.0.0.0-beta1,<2.3.12", "ezsystems/ezplatform-kernel": "<1.2.5.1-dev|>=1.3,<1.3.35", "ezsystems/ezplatform-rest": ">=1.2,<=1.2.2|>=1.3,<1.3.8", - "ezsystems/ezplatform-richtext": ">=2.3,<2.3.7.1-dev", + "ezsystems/ezplatform-richtext": ">=2.3,<2.3.7.1-dev|>=3.3,<3.3.40", "ezsystems/ezplatform-solr-search-engine": ">=1.7,<1.7.12|>=2,<2.0.2|>=3.3,<3.3.15", "ezsystems/ezplatform-user": ">=1,<1.0.1", "ezsystems/ezpublish-kernel": "<6.13.8.2-dev|>=7,<7.5.31", "ezsystems/ezpublish-legacy": "<=2017.12.7.3|>=2018.6,<=2019.03.5.1", "ezsystems/platform-ui-assets-bundle": ">=4.2,<4.2.3", "ezsystems/repository-forms": ">=2.3,<2.3.2.1-dev|>=2.5,<2.5.15", - "ezyang/htmlpurifier": "<4.1.1", + "ezyang/htmlpurifier": "<=4.2", "facade/ignition": "<1.16.15|>=2,<2.4.2|>=2.5,<2.5.2", "facturascripts/facturascripts": "<=2022.08", "fastly/magento2": "<1.2.26", "feehi/cms": "<=2.1.1", "feehi/feehicms": "<=2.1.1", "fenom/fenom": "<=2.12.1", + "filament/actions": ">=3.2,<3.2.123", + "filament/infolists": ">=3,<3.2.115", + "filament/tables": ">=3,<3.2.115", "filegator/filegator": "<7.8", "filp/whoops": "<2.1.13", "fineuploader/php-traditional-server": "<=1.2.2", "firebase/php-jwt": "<6", + "fisharebest/webtrees": "<=2.1.18", "fixpunkt/fp-masterquiz": "<2.2.1|>=3,<3.5.2", "fixpunkt/fp-newsletter": "<1.1.1|>=2,<2.1.2|>=2.2,<3.2.6", "flarum/core": "<1.8.5", @@ -3617,19 +3943,19 @@ "friendsofsymfony/rest-bundle": ">=1.2,<1.2.2", "friendsofsymfony/user-bundle": ">=1,<1.3.5", "friendsofsymfony1/swiftmailer": ">=4,<5.4.13|>=6,<6.2.5", - "friendsofsymfony1/symfony1": ">=1.1,<1.15.19", + "friendsofsymfony1/symfony1": ">=1.1,<1.5.19", "friendsoftypo3/mediace": ">=7.6.2,<7.6.5", "friendsoftypo3/openid": ">=4.5,<4.5.31|>=4.7,<4.7.16|>=6,<6.0.11|>=6.1,<6.1.6", "froala/wysiwyg-editor": "<3.2.7|>=4.0.1,<=4.1.3", - "froxlor/froxlor": "<2.1.9", + "froxlor/froxlor": "<=2.2.0.0-RC3", "frozennode/administrator": "<=5.0.12", "fuel/core": "<1.8.1", - "funadmin/funadmin": "<=3.2|>=3.3.2,<=3.3.3", + "funadmin/funadmin": "<=5.0.2", "gaoming13/wechat-php-sdk": "<=1.10.2", "genix/cms": "<=1.1.11", "getformwork/formwork": "<1.13.1|==2.0.0.0-beta1", "getgrav/grav": "<1.7.46", - "getkirby/cms": "<4.1.1", + "getkirby/cms": "<=3.6.6.5|>=3.7,<=3.7.5.4|>=3.8,<=3.8.4.3|>=3.9,<=3.9.8.1|>=3.10,<=3.10.1|>=4,<=4.3", "getkirby/kirby": "<=2.5.12", "getkirby/panel": "<2.5.14", "getkirby/starterkit": "<=3.7.0.2", @@ -3655,8 +3981,9 @@ "hov/jobfair": "<1.0.13|>=2,<2.0.2", "httpsoft/http-message": "<1.0.12", "hyn/multi-tenant": ">=5.6,<5.7.2", - "ibexa/admin-ui": ">=4.2,<4.2.3", + "ibexa/admin-ui": ">=4.2,<4.2.3|>=4.6.0.0-beta1,<4.6.9", "ibexa/core": ">=4,<4.0.7|>=4.1,<4.1.4|>=4.2,<4.2.3|>=4.5,<4.5.6|>=4.6,<4.6.2", + "ibexa/fieldtype-richtext": ">=4.6,<4.6.10", "ibexa/graphql": ">=2.5,<2.5.31|>=3.3,<3.3.28|>=4.2,<4.2.3", "ibexa/post-install": "<=1.0.4", "ibexa/solr": ">=4.5,<4.5.4", @@ -3675,9 +4002,11 @@ "in2code/femanager": "<5.5.3|>=6,<6.3.4|>=7,<7.2.3", "in2code/ipandlanguageredirect": "<5.1.2", "in2code/lux": "<17.6.1|>=18,<24.0.2", + "in2code/powermail": "<7.5.1|>=8,<8.5.1|>=9,<10.9.1|>=11,<12.4.1", "innologi/typo3-appointments": "<2.0.6", "intelliants/subrion": "<4.2.2", "inter-mediator/inter-mediator": "==5.5", + "ipl/web": "<0.10.1", "islandora/islandora": ">=2,<2.4.1", "ivankristianto/phpwhois": "<=4.3", "jackalope/jackalope-doctrine-dbal": "<1.7.4", @@ -3703,21 +4032,24 @@ "kelvinmo/simplexrd": "<3.1.1", "kevinpapst/kimai2": "<1.16.7", "khodakhah/nodcms": "<=3", - "kimai/kimai": "<2.16", + "kimai/kimai": "<=2.20.1", "kitodo/presentation": "<3.2.3|>=3.3,<3.3.4", "klaviyo/magento2-extension": ">=1,<3", "knplabs/knp-snappy": "<=1.4.2", "kohana/core": "<3.3.3", - "krayin/laravel-crm": "<1.2.2", + "krayin/laravel-crm": "<=1.3", "kreait/firebase-php": ">=3.2,<3.8.1", "kumbiaphp/kumbiapp": "<=1.1.1", "la-haute-societe/tcpdf": "<6.2.22", "laminas/laminas-diactoros": "<2.18.1|==2.19|==2.20|==2.21|==2.22|==2.23|>=2.24,<2.24.2|>=2.25,<2.25.2", "laminas/laminas-form": "<2.17.1|>=3,<3.0.2|>=3.1,<3.1.1", "laminas/laminas-http": "<2.14.2", + "lara-zeus/artemis": ">=1,<=1.0.6", + "lara-zeus/dynamic-dashboard": ">=3,<=3.0.1", "laravel/fortify": "<1.11.1", "laravel/framework": "<6.20.44|>=7,<7.30.6|>=8,<8.75", "laravel/laravel": ">=5.4,<5.4.22", + "laravel/reverb": "<1.4", "laravel/socialite": ">=1,<2.0.10", "latte/latte": "<2.10.8", "lavalite/cms": "<=9|==10.1", @@ -3730,26 +4062,30 @@ "librenms/librenms": "<2017.08.18", "liftkit/database": "<2.13.2", "lightsaml/lightsaml": "<1.3.5", - "limesurvey/limesurvey": "<3.27.19", + "limesurvey/limesurvey": "<6.5.12", "livehelperchat/livehelperchat": "<=3.91", - "livewire/livewire": ">2.2.4,<2.2.6|>=3.3.5,<3.4.9", + "livewire/livewire": "<2.12.7|>=3.0.0.0-beta1,<3.5.2", "lms/routes": "<2.1.1", "localizationteam/l10nmgr": "<7.4|>=8,<8.7|>=9,<9.2", "luyadev/yii-helpers": "<1.2.1", - "magento/community-edition": "<2.4.5|==2.4.5|>=2.4.5.0-patch1,<2.4.5.0-patch8|==2.4.6|>=2.4.6.0-patch1,<2.4.6.0-patch6|==2.4.7", + "maestroerror/php-heic-to-jpg": "<1.0.5", + "magento/community-edition": "<2.4.5|==2.4.5|>=2.4.5.0-patch1,<2.4.5.0-patch10|==2.4.6|>=2.4.6.0-patch1,<2.4.6.0-patch8|>=2.4.7.0-beta1,<2.4.7.0-patch3", "magento/core": "<=1.9.4.5", "magento/magento1ce": "<1.9.4.3-dev", "magento/magento1ee": ">=1,<1.14.4.3-dev", - "magento/product-community-edition": ">=2,<2.2.10|>=2.3,<2.3.2.0-patch2", + "magento/product-community-edition": "<2.4.4.0-patch9|>=2.4.5,<2.4.5.0-patch8|>=2.4.6,<2.4.6.0-patch6|>=2.4.7,<2.4.7.0-patch1", "magneto/core": "<1.9.4.4-dev", "maikuolan/phpmussel": ">=1,<1.6", "mainwp/mainwp": "<=4.4.3.3", - "mantisbt/mantisbt": "<2.26.2", + "mantisbt/mantisbt": "<=2.26.3", "marcwillmann/turn": "<0.3.3", "matyhtf/framework": "<3.0.6", - "mautic/core": "<4.4.12|>=5.0.0.0-alpha,<5.0.4", + "mautic/core": "<4.4.13|>=5,<5.1.1", + "mautic/core-lib": ">=1.0.0.0-beta,<4.4.13|>=5.0.0.0-alpha,<5.1.1", + "maximebf/debugbar": "<1.19", "mdanter/ecc": "<2", - "mediawiki/core": "<1.36.2", + "mediawiki/cargo": "<3.6.1", + "mediawiki/core": "<1.39.5|==1.40", "mediawiki/matomo": "<2.4.3", "mediawiki/semantic-media-wiki": "<4.0.2", "melisplatform/melis-asset-manager": "<5.0.1", @@ -3760,7 +4096,7 @@ "microsoft/microsoft-graph": ">=1.16,<1.109.1|>=2,<2.0.1", "microsoft/microsoft-graph-beta": "<2.0.1", "microsoft/microsoft-graph-core": "<2.0.2", - "microweber/microweber": "<=2.0.4", + "microweber/microweber": "<=2.0.16", "mikehaertl/php-shellcommand": "<1.6.1", "miniorange/miniorange-saml": "<1.4.3", "mittwald/typo3_forum": "<1.2.1", @@ -3769,7 +4105,7 @@ "mojo42/jirafeau": "<4.4", "mongodb/mongodb": ">=1,<1.9.2", "monolog/monolog": ">=1.8,<1.12", - "moodle/moodle": "<4.3.5|>=4.4.0.0-beta,<4.4.1", + "moodle/moodle": "<4.3.6|>=4.4.0.0-beta,<4.4.2", "mos/cimage": "<0.7.19", "movim/moxl": ">=0.8,<=0.10", "movingbytes/social-network": "<=1.2.1", @@ -3782,6 +4118,7 @@ "munkireport/softwareupdate": "<1.6", "mustache/mustache": ">=2,<2.14.1", "namshi/jose": "<2.2", + "nategood/httpful": "<1", "neoan3-apps/template": "<1.1.1", "neorazorx/facturascripts": "<2022.04", "neos/flow": ">=1,<1.0.4|>=1.1,<1.1.1|>=2,<2.0.1|>=2.3,<2.3.16|>=3,<3.0.12|>=3.1,<3.1.10|>=3.2,<3.2.13|>=3.3,<3.3.13|>=4,<4.0.6", @@ -3804,16 +4141,16 @@ "nzo/url-encryptor-bundle": ">=4,<4.3.2|>=5,<5.0.1", "october/backend": "<1.1.2", "october/cms": "<1.0.469|==1.0.469|==1.0.471|==1.1.1", - "october/october": "<=3.4.4", + "october/october": "<=3.6.4", "october/rain": "<1.0.472|>=1.1,<1.1.2", "october/system": "<1.0.476|>=1.1,<1.1.12|>=2,<2.2.34|>=3,<3.5.15", "omeka/omeka-s": "<4.0.3", "onelogin/php-saml": "<2.10.4", "oneup/uploader-bundle": ">=1,<1.9.3|>=2,<2.1.5", "open-web-analytics/open-web-analytics": "<1.7.4", - "opencart/opencart": "<=3.0.3.9|>=4", + "opencart/opencart": ">=0", "openid/php-openid": "<2.3", - "openmage/magento-lts": "<20.5", + "openmage/magento-lts": "<20.10.1", "opensolutions/vimbadmin": "<=3.0.15", "opensource-workshop/connect-cms": "<1.7.2|>=2,<2.3.2", "orchid/platform": ">=9,<9.4.4|>=14.0.0.0-alpha4,<14.5", @@ -3823,6 +4160,7 @@ "oro/crm-call-bundle": ">=4.2,<=4.2.5|>=5,<5.0.4|>=5.1,<5.1.1", "oro/customer-portal": ">=4.1,<=4.1.13|>=4.2,<=4.2.10|>=5,<=5.0.11|>=5.1,<=5.1.3", "oro/platform": ">=1.7,<1.7.4|>=3.1,<3.1.29|>=4.1,<4.1.17|>=4.2,<=4.2.10|>=5,<=5.0.12|>=5.1,<=5.1.3", + "oveleon/contao-cookiebar": "<1.16.3|>=2,<2.1.3", "oxid-esales/oxideshop-ce": "<4.5", "oxid-esales/paymorrow-module": ">=1,<1.0.2|>=2,<2.0.1", "packbackbooks/lti-1-3-php-library": "<5", @@ -3846,7 +4184,7 @@ "phenx/php-svg-lib": "<0.5.2", "php-censor/php-censor": "<2.0.13|>=2.1,<2.1.5", "php-mod/curl": "<2.3.2", - "phpbb/phpbb": "<3.2.10|>=3.3,<3.3.1", + "phpbb/phpbb": "<3.3.11", "phpems/phpems": ">=6,<=6.1.3", "phpfastcache/phpfastcache": "<6.1.5|>=7,<7.1.2|>=8,<8.0.7", "phpmailer/phpmailer": "<6.5", @@ -3854,8 +4192,8 @@ "phpmyadmin/phpmyadmin": "<5.2.1", "phpmyfaq/phpmyfaq": "<3.2.5|==3.2.5", "phpoffice/common": "<0.2.9", - "phpoffice/phpexcel": "<1.8", - "phpoffice/phpspreadsheet": "<1.16", + "phpoffice/phpexcel": "<1.8.1", + "phpoffice/phpspreadsheet": "<1.29.2|>=2,<2.1.1|>=2.2,<2.3", "phpseclib/phpseclib": "<2.0.47|>=3,<3.0.36", "phpservermon/phpservermon": "<3.6", "phpsysinfo/phpsysinfo": "<3.4.3", @@ -3864,9 +4202,10 @@ "phpxmlrpc/extras": "<0.6.1", "phpxmlrpc/phpxmlrpc": "<4.9.2", "pi/pi": "<=2.5", - "pimcore/admin-ui-classic-bundle": "<=1.4.2", + "pimcore/admin-ui-classic-bundle": "<1.5.4", "pimcore/customer-management-framework-bundle": "<4.0.6", "pimcore/data-hub": "<1.2.4", + "pimcore/data-importer": "<1.8.9|>=1.9,<1.9.3", "pimcore/demo": "<10.3", "pimcore/ecommerce-framework-bundle": "<1.0.10", "pimcore/perspective-editor": "<1.5.1", @@ -3888,15 +4227,16 @@ "prestashop/ps_facetedsearch": "<3.4.1", "prestashop/ps_linklist": "<3.1", "privatebin/privatebin": "<1.4|>=1.5,<1.7.4", - "processwire/processwire": "<=3.0.210", + "processwire/processwire": "<=3.0.229", "propel/propel": ">=2.0.0.0-alpha1,<=2.0.0.0-alpha7", "propel/propel1": ">=1,<=1.7.1", - "pterodactyl/panel": "<1.11.6", + "pterodactyl/panel": "<1.11.8", "ptheofan/yii2-statemachine": ">=2.0.0.0-RC1-dev,<=2", "ptrofimov/beanstalk_console": "<1.7.14", "pubnub/pubnub": "<6.1", "pusher/pusher-php-server": "<2.2.1", "pwweb/laravel-core": "<=0.3.6.0-beta", + "pxlrbt/filament-excel": "<1.1.14|>=2.0.0.0-alpha,<2.3.3", "pyrocms/pyrocms": "<=3.9.1", "qcubed/qcubed": "<=3.1.1", "quickapps/cms": "<=2.0.0.0-beta2", @@ -3907,7 +4247,7 @@ "rap2hpoutre/laravel-log-viewer": "<0.13", "react/http": ">=0.7,<1.9", "really-simple-plugins/complianz-gdpr": "<6.4.2", - "redaxo/source": "<=5.15.1", + "redaxo/source": "<=5.17.1", "remdex/livehelperchat": "<4.29", "reportico-web/reportico": "<=8.1", "rhukster/dom-sanitizer": "<1.0.7", @@ -3924,8 +4264,8 @@ "serluck/phpwhois": "<=4.2.6", "sfroemken/url_redirect": "<=1.2.1", "sheng/yiicms": "<=1.2", - "shopware/core": "<6.5.8.8-dev|>=6.6.0.0-RC1-dev,<6.6.1", - "shopware/platform": "<6.5.8.8-dev|>=6.6.0.0-RC1-dev,<6.6.1", + "shopware/core": "<=6.5.8.12|>=6.6,<=6.6.5", + "shopware/platform": "<=6.5.8.12|>=6.6,<=6.6.5", "shopware/production": "<=6.3.5.2", "shopware/shopware": "<=5.7.17", "shopware/storefront": "<=6.4.8.1|>=6.5.8,<6.5.8.7-dev", @@ -3937,11 +4277,12 @@ "silverstripe/cms": "<4.11.3", "silverstripe/comments": ">=1.3,<3.1.1", "silverstripe/forum": "<=0.6.1|>=0.7,<=0.7.3", - "silverstripe/framework": "<4.13.39|>=5,<5.1.11", + "silverstripe/framework": "<5.2.16", "silverstripe/graphql": ">=2,<2.0.5|>=3,<3.8.2|>=4,<4.3.7|>=5,<5.1.3", "silverstripe/hybridsessions": ">=1,<2.4.1|>=2.5,<2.5.1", "silverstripe/recipe-cms": ">=4.5,<4.5.3", "silverstripe/registry": ">=2.1,<2.1.2|>=2.2,<2.2.1", + "silverstripe/reports": "<5.2.3", "silverstripe/restfulserver": ">=1,<1.0.9|>=2,<2.0.4|>=2.1,<2.1.2", "silverstripe/silverstripe-omnipay": "<2.5.2|>=3,<3.0.2|>=3.1,<3.1.4|>=3.2,<3.2.1", "silverstripe/subsites": ">=2,<2.6.1", @@ -3962,7 +4303,7 @@ "slim/slim": "<2.6", "slub/slub-events": "<3.0.3", "smarty/smarty": "<4.5.3|>=5,<5.1.1", - "snipe/snipe-it": "<6.4.2", + "snipe/snipe-it": "<7.0.10", "socalnick/scn-social-auth": "<1.15.2", "socialiteproviders/steam": "<1.1", "spatie/browsershot": "<3.57.4", @@ -3972,14 +4313,15 @@ "spoonity/tcpdf": "<6.2.22", "squizlabs/php_codesniffer": ">=1,<2.8.1|>=3,<3.0.1", "ssddanbrown/bookstack": "<24.05.1", + "starcitizentools/citizen-skin": ">=2.6.3,<2.31", "statamic/cms": "<4.46|>=5.3,<5.6.2", "stormpath/sdk": "<9.9.99", - "studio-42/elfinder": "<2.1.62", + "studio-42/elfinder": "<=2.1.64", "studiomitte/friendlycaptcha": "<0.1.4", "subhh/libconnect": "<7.0.8|>=8,<8.1", "sukohi/surpass": "<1", "sulu/form-bundle": ">=2,<2.5.3", - "sulu/sulu": "<1.6.44|>=2,<2.4.17|>=2.5,<2.5.13", + "sulu/sulu": "<1.6.44|>=2,<2.5.21|>=2.6,<2.6.5", "sumocoders/framework-user-bundle": "<1.4", "superbig/craft-audit": "<3.0.2", "swag/paypal": "<5.4.4", @@ -3990,7 +4332,7 @@ "sylius/grid-bundle": "<1.10.1", "sylius/paypal-plugin": ">=1,<1.2.4|>=1.3,<1.3.1", "sylius/resource-bundle": ">=1,<1.3.14|>=1.4,<1.4.7|>=1.5,<1.5.2|>=1.6,<1.6.4", - "sylius/sylius": "<1.9.10|>=1.10,<1.10.11|>=1.11,<1.11.2|>=1.12.0.0-alpha1,<1.12.16|>=1.13.0.0-alpha1,<1.13.1", + "sylius/sylius": "<1.12.19|>=1.13.0.0-alpha1,<1.13.4", "symbiote/silverstripe-multivaluefield": ">=3,<3.1", "symbiote/silverstripe-queuedjobs": ">=3,<3.0.2|>=3.1,<3.1.4|>=4,<4.0.7|>=4.1,<4.1.2|>=4.2,<4.2.4|>=4.3,<4.3.3|>=4.4,<4.4.3|>=4.5,<4.5.1|>=4.6,<4.6.4", "symbiote/silverstripe-seed": "<6.0.3", @@ -4001,7 +4343,8 @@ "symfony/error-handler": ">=4.4,<4.4.4|>=5,<5.0.4", "symfony/form": ">=2.3,<2.3.35|>=2.4,<2.6.12|>=2.7,<2.7.50|>=2.8,<2.8.49|>=3,<3.4.20|>=4,<4.0.15|>=4.1,<4.1.9|>=4.2,<4.2.1", "symfony/framework-bundle": ">=2,<2.3.18|>=2.4,<2.4.8|>=2.5,<2.5.2|>=2.7,<2.7.51|>=2.8,<2.8.50|>=3,<3.4.26|>=4,<4.1.12|>=4.2,<4.2.7|>=5.3.14,<5.3.15|>=5.4.3,<5.4.4|>=6.0.3,<6.0.4", - "symfony/http-foundation": ">=2,<2.8.52|>=3,<3.4.35|>=4,<4.2.12|>=4.3,<4.3.8|>=4.4,<4.4.7|>=5,<5.0.7", + "symfony/http-client": ">=4.3,<5.4.46|>=6,<6.4.14|>=7,<7.1.7", + "symfony/http-foundation": "<5.4.46|>=6,<6.4.14|>=7,<7.1.7", "symfony/http-kernel": ">=2,<4.4.50|>=5,<5.4.20|>=6,<6.0.20|>=6.1,<6.1.12|>=6.2,<6.2.6", "symfony/intl": ">=2.7,<2.7.38|>=2.8,<2.8.31|>=3,<3.2.14|>=3.3,<3.3.13", "symfony/maker-bundle": ">=1.27,<1.29.2|>=1.30,<1.31.1", @@ -4009,20 +4352,22 @@ "symfony/phpunit-bridge": ">=2.8,<2.8.50|>=3,<3.4.26|>=4,<4.1.12|>=4.2,<4.2.7", "symfony/polyfill": ">=1,<1.10", "symfony/polyfill-php55": ">=1,<1.10", + "symfony/process": "<5.4.46|>=6,<6.4.14|>=7,<7.1.7", "symfony/proxy-manager-bridge": ">=2.7,<2.7.51|>=2.8,<2.8.50|>=3,<3.4.26|>=4,<4.1.12|>=4.2,<4.2.7", "symfony/routing": ">=2,<2.0.19", + "symfony/runtime": ">=5.3,<5.4.46|>=6,<6.4.14|>=7,<7.1.7", "symfony/security": ">=2,<2.7.51|>=2.8,<3.4.49|>=4,<4.4.24|>=5,<5.2.8", - "symfony/security-bundle": ">=2,<4.4.50|>=5,<5.4.20|>=6,<6.0.20|>=6.1,<6.1.12|>=6.2,<6.2.6", + "symfony/security-bundle": ">=2,<4.4.50|>=5,<5.4.20|>=6,<6.0.20|>=6.1,<6.1.12|>=6.2,<6.4.10|>=7,<7.0.10|>=7.1,<7.1.3", "symfony/security-core": ">=2.4,<2.6.13|>=2.7,<2.7.9|>=2.7.30,<2.7.32|>=2.8,<3.4.49|>=4,<4.4.24|>=5,<5.2.9", "symfony/security-csrf": ">=2.4,<2.7.48|>=2.8,<2.8.41|>=3,<3.3.17|>=3.4,<3.4.11|>=4,<4.0.11", "symfony/security-guard": ">=2.8,<3.4.48|>=4,<4.4.23|>=5,<5.2.8", "symfony/security-http": ">=2.3,<2.3.41|>=2.4,<2.7.51|>=2.8,<2.8.50|>=3,<3.4.26|>=4,<4.2.12|>=4.3,<4.3.8|>=4.4,<4.4.7|>=5,<5.0.7|>=5.1,<5.2.8|>=5.3,<5.3.2|>=5.4,<5.4.31|>=6,<6.3.8", "symfony/serializer": ">=2,<2.0.11|>=4.1,<4.4.35|>=5,<5.3.12", - "symfony/symfony": ">=2,<4.4.51|>=5,<5.4.31|>=6,<6.3.8", + "symfony/symfony": "<5.4.46|>=6,<6.4.14|>=7,<7.1.7", "symfony/translation": ">=2,<2.0.17", "symfony/twig-bridge": ">=2,<4.4.51|>=5,<5.4.31|>=6,<6.3.8", "symfony/ux-autocomplete": "<2.11.2", - "symfony/validator": ">=2,<2.0.24|>=2.1,<2.1.12|>=2.2,<2.2.5|>=2.3,<2.3.3", + "symfony/validator": "<5.4.43|>=6,<6.4.11|>=7,<7.1.4", "symfony/var-exporter": ">=4.2,<4.2.12|>=4.3,<4.3.8", "symfony/web-profiler-bundle": ">=2,<2.3.19|>=2.4,<2.4.9|>=2.5,<2.5.4", "symfony/webhook": ">=6.3,<6.3.8", @@ -4046,17 +4391,18 @@ "tinymighty/wiki-seo": "<1.2.2", "titon/framework": "<9.9.99", "tobiasbg/tablepress": "<=2.0.0.0-RC1", - "topthink/framework": "<6.0.17|>=6.1,<6.1.5|>=8,<8.0.4", + "topthink/framework": "<6.0.17|>=6.1,<=8.0.4", "topthink/think": "<=6.1.1", - "topthink/thinkphp": "<=3.2.3", + "topthink/thinkphp": "<=3.2.3|>=6.1.3,<=8.0.4", "torrentpier/torrentpier": "<=2.4.3", "tpwd/ke_search": "<4.0.3|>=4.1,<4.6.6|>=5,<5.0.2", - "tribalsystems/zenario": "<9.5.60602", + "tribalsystems/zenario": "<=9.7.61188", "truckersmp/phpwhois": "<=4.3.1", "ttskch/pagination-service-provider": "<1", - "twig/twig": "<1.44.7|>=2,<2.15.3|>=3,<3.4.3", + "twbs/bootstrap": "<=3.4.1|>=4,<=4.6.2", + "twig/twig": "<3.11.2|>=3.12,<3.14.1", "typo3/cms": "<9.5.29|>=10,<10.4.35|>=11,<11.5.23|>=12,<12.2", - "typo3/cms-backend": "<4.1.14|>=4.2,<4.2.15|>=4.3,<4.3.7|>=4.4,<4.4.4|>=7,<=7.6.50|>=8,<=8.7.39|>=9,<=9.5.24|>=10,<=10.4.13|>=11,<=11.1", + "typo3/cms-backend": "<4.1.14|>=4.2,<4.2.15|>=4.3,<4.3.7|>=4.4,<4.4.4|>=7,<=7.6.50|>=8,<=8.7.39|>=9,<=9.5.24|>=10,<10.4.46|>=11,<11.5.40|>=12,<12.4.21|>=13,<13.3.1", "typo3/cms-core": "<=8.7.56|>=9,<=9.5.47|>=10,<=10.4.44|>=11,<=11.5.36|>=12,<=12.4.14|>=13,<=13.1", "typo3/cms-extbase": "<6.2.24|>=7,<7.6.8|==8.1.1", "typo3/cms-fluid": "<4.3.4|>=4.4,<4.4.1", @@ -4073,6 +4419,7 @@ "ua-parser/uap-php": "<3.8", "uasoft-indonesia/badaso": "<=2.9.7", "unisharp/laravel-filemanager": "<2.6.4", + "unopim/unopim": "<0.1.4", "userfrosting/userfrosting": ">=0.3.1,<4.6.3", "usmanhalalit/pixie": "<1.0.3|>=2,<2.0.2", "uvdesk/community-skeleton": "<=1.1.1", @@ -4106,6 +4453,7 @@ "winter/wn-dusk-plugin": "<2.1", "winter/wn-system-module": "<1.2.4", "wintercms/winter": "<=1.2.3", + "wireui/wireui": "<1.19.3|>=2,<2.1.3", "woocommerce/woocommerce": "<6.6|>=8.8,<8.8.5|>=8.9,<8.9.3", "wp-cli/wp-cli": ">=0.12,<2.5", "wp-graphql/wp-graphql": "<=1.14.5", @@ -4117,7 +4465,7 @@ "xataface/xataface": "<3", "xpressengine/xpressengine": "<3.0.15", "yab/quarx": "<2.4.5", - "yeswiki/yeswiki": "<4.1", + "yeswiki/yeswiki": "<=4.4.4", "yetiforce/yetiforce-crm": "<=6.4", "yidashi/yii2cmf": "<=2", "yii2mod/yii2-cms": "<1.9.2", @@ -4208,7 +4556,7 @@ "type": "tidelift" } ], - "time": "2024-07-15T18:05:53+00:00" + "time": "2024-11-07T19:04:57+00:00" } ], "aliases": [], diff --git a/lib/Db/PublicationType.php b/lib/Db/PublicationType.php index 510a1e6c..68b2e6e7 100644 --- a/lib/Db/PublicationType.php +++ b/lib/Db/PublicationType.php @@ -5,6 +5,7 @@ use DateTime; use JsonSerializable; use OCP\AppFramework\Db\Entity; +use OCP\IURLGenerator; class PublicationType extends Entity implements JsonSerializable { @@ -122,4 +123,33 @@ public function jsonSerialize(): array return $array; } + + /** + * Generate a JSON-Schema definition for the data field of a publication. + * + * @param IURLGenerator $urlGenerator An URL generator to generate the identifier of the schema. + * + * @return object The JSON-Schema object defining the data field of a publication. + */ + public function getSchema(IURLGenerator $urlGenerator): object + { + $schema = []; + $schema['$schema'] = 'https://json-schema.org/draft/2020-12/schema'; + $schema['$id'] = $urlGenerator->getAbsoluteURL($urlGenerator->linkToRoute('opencatalogi.publication_types.show', ['id' => $this->getUuid()])); + $schema['type'] = 'object'; + $schema['required'] = []; + $schema['properties'] = []; + + foreach ($this->getProperties() as $name => $property) { + if ($property['required'] === true) { + $schema['required'][] = $name; + } + unset($property['title'], $property['required']); + + // Remove empty fields with array_filter(), and add it to the properties of the schema. + $schema['properties'][$name] = array_filter($property); + } + + return json_decode(json_encode($schema)); + } } diff --git a/lib/Service/ValidationService.php b/lib/Service/ValidationService.php index 38371705..0a28ed7a 100644 --- a/lib/Service/ValidationService.php +++ b/lib/Service/ValidationService.php @@ -15,6 +15,7 @@ use Opis\JsonSchema\Validator; use Psr\Container\ContainerExceptionInterface; use Psr\Container\NotFoundExceptionInterface; +use Symfony\Component\Uid\Uuid; /** * Class ValidationService @@ -49,6 +50,10 @@ public function validatePublication(array $publication): array $publicationType = (new PublicationType())->hydrate($publicationType); + if(Uuid::isValid($publicationTypeId)) { + $publicationType->setUuid($publicationTypeId); + } + $validator = new Validator(); $validator->setMaxErrors(100); From a018dadae24a7841484c2782100b66f978897b5d Mon Sep 17 00:00:00 2001 From: Ruben van der Linde Date: Tue, 19 Nov 2024 15:55:21 +0100 Subject: [PATCH 05/18] phpcs --- phpcs.xml | 154 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 154 insertions(+) create mode 100644 phpcs.xml diff --git a/phpcs.xml b/phpcs.xml new file mode 100644 index 00000000..eb935f62 --- /dev/null +++ b/phpcs.xml @@ -0,0 +1,154 @@ + + + The coding standard for PHP_CodeSniffer itself, for more config -> for https://github.com/squizlabs/PHP_CodeSniffer/wiki/Configuration-Options. + + README.md + src + + + + + + + + error + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 0 + + + 0 + + + 0 + + + 0 + + + + + + 0 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + error + + + + + error + + + + + tests/bootstrap\.php + + + + + tests/Core/Tokenizer/StableCommentWhitespaceWinTest\.php + + + From c0d23d657b5c880dc8143ce0e631023e187c15bc Mon Sep 17 00:00:00 2001 From: Barry Brands Date: Thu, 21 Nov 2024 10:46:39 +0100 Subject: [PATCH 06/18] Fix summary --- src/sidebars/directory/DirectorySideBar.vue | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/sidebars/directory/DirectorySideBar.vue b/src/sidebars/directory/DirectorySideBar.vue index 55cdb449..849c394d 100644 --- a/src/sidebars/directory/DirectorySideBar.vue +++ b/src/sidebars/directory/DirectorySideBar.vue @@ -46,7 +46,7 @@ import { navigationStore, directoryStore, publicationTypeStore } from '../../sto
Samenvatting: - {{ directoryStore.listingItem?.summery }} + {{ directoryStore.listingItem?.summary }}
Status: From 76bba425b9f81ea8a3758fabecd88e4439c270bc Mon Sep 17 00:00:00 2001 From: Barry Brands Date: Thu, 21 Nov 2024 11:47:59 +0100 Subject: [PATCH 07/18] Change validation object and set default values --- lib/Service/ValidationService.php | 20 +++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/lib/Service/ValidationService.php b/lib/Service/ValidationService.php index 0a28ed7a..988f42a1 100644 --- a/lib/Service/ValidationService.php +++ b/lib/Service/ValidationService.php @@ -57,17 +57,31 @@ public function validatePublication(array $publication): array $validator = new Validator(); $validator->setMaxErrors(100); - if(empty($publicationType->getProperties()) === true) { + if (empty($publicationType->getProperties()) === true) { return $publication; } + + // Check for default values, and only set the property if the property is empty + foreach ($publicationType->getProperties() as $property) { + if (isset($property['default']) === true && empty($property['default']) === false && (isset($publication['data'][$property['title']]) === false || empty($publication['data'][$property['title']]) === true)) { + $publication['data'][$property['title']] = $property['default']; + } + } + $result = $validator->validate(data: (object) json_decode(json_encode($publication['data'])), schema: $publicationType->getSchema($this->urlGenerator)); - $publication['validation'] = []; + $publication['validation'] = [ + 'errors' => [], + 'valid' => true + ]; if ($result->hasError()) { $errors = (new ErrorFormatter())->format($result->error()); - $publication['validation'] = $errors; + foreach ($errors as $error) { + $publication['validation']['errors'][] = $error[0]; + } + $publication['validation']['valid'] = false; } return $publication; From d1416cd0f584e9967fc34232039b1a5b39208d5b Mon Sep 17 00:00:00 2001 From: Barry Brands Date: Thu, 21 Nov 2024 13:17:17 +0100 Subject: [PATCH 08/18] Small cleanup --- lib/Service/ValidationService.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/Service/ValidationService.php b/lib/Service/ValidationService.php index 988f42a1..649f369d 100644 --- a/lib/Service/ValidationService.php +++ b/lib/Service/ValidationService.php @@ -50,7 +50,7 @@ public function validatePublication(array $publication): array $publicationType = (new PublicationType())->hydrate($publicationType); - if(Uuid::isValid($publicationTypeId)) { + if (Uuid::isValid($publicationTypeId) === true) { $publicationType->setUuid($publicationTypeId); } From ccce012818288191444a42e698050459e695d58c Mon Sep 17 00:00:00 2001 From: Barry Brands Date: Thu, 21 Nov 2024 15:49:04 +0100 Subject: [PATCH 09/18] Get PublicationType and ValidationService more inline with OpenRegister --- lib/Db/PublicationType.php | 95 ++++++++++++++----------------- lib/Service/ValidationService.php | 2 +- 2 files changed, 45 insertions(+), 52 deletions(-) diff --git a/lib/Db/PublicationType.php b/lib/Db/PublicationType.php index 68b2e6e7..61795df3 100644 --- a/lib/Db/PublicationType.php +++ b/lib/Db/PublicationType.php @@ -13,11 +13,11 @@ class PublicationType extends Entity implements JsonSerializable protected ?string $version = '0.0.1'; protected ?string $title = null; protected ?string $description = null; - protected ?array $required = null; - protected ?array $properties = null; + protected ?array $required = []; + protected ?array $properties = []; protected ?string $source = null; protected ?string $summary = null; - protected ?array $archive = null; + protected ?array $archive = []; protected ?DateTime $updated = null; protected ?DateTime $created = null; @@ -48,14 +48,6 @@ public function hydrate(array $object): self { $jsonFields = $this->getJsonFields(); - // Remove any fields that start with an underscore - // These are typically internal fields that shouldn't be updated directly - foreach ($object as $key => $value) { - if (str_starts_with($key, '_')) { - unset($object[$key]); - } - } - foreach ($object as $key => $value) { if (in_array($key, $jsonFields) === true && $value === []) { $value = null; @@ -73,28 +65,23 @@ public function hydrate(array $object): self return $this; } + /** + * Serializes the schema to an array + * + * @return array + */ public function jsonSerialize(): array { - $properties = []; - foreach ($this->properties ?? [] as $key => $property) { - $properties[$key] = $property; - if (isset($property['type']) === false) { - $properties[$key] = $property; - continue; - } - switch ($property['format'] ?? '') { - case 'string': - case 'array': - $properties[$key]['default'] = (string) ($property['default'] ?? ''); - break; - case 'int': - case 'integer': - case 'number': - $properties[$key]['default'] = (int) ($property['default'] ?? 0); - break; - case 'bool': - $properties[$key]['default'] = (bool) ($property['default'] ?? false); - break; + $required = $this->required ?? []; + $properties = []; + if (isset($this->properties) === true) { + foreach ($this->properties as $key => $property) { + $title = $property['title'] ?? $key; + if ($property['required'] === true && in_array($title, $required) === false) { + $required[] = $title; + } + + $properties[$title] = $property; } } @@ -104,7 +91,7 @@ public function jsonSerialize(): array 'version' => $this->version, 'title' => $this->title, 'description' => $this->description, - 'required' => $this->required, + 'required' => $required, 'properties' => $properties, 'source' => $this->source, 'summary' => $this->summary, @@ -131,25 +118,31 @@ public function jsonSerialize(): array * * @return object The JSON-Schema object defining the data field of a publication. */ - public function getSchema(IURLGenerator $urlGenerator): object + public function getSchemaObject(IURLGenerator $urlGenerator): object { - $schema = []; - $schema['$schema'] = 'https://json-schema.org/draft/2020-12/schema'; - $schema['$id'] = $urlGenerator->getAbsoluteURL($urlGenerator->linkToRoute('opencatalogi.publication_types.show', ['id' => $this->getUuid()])); - $schema['type'] = 'object'; - $schema['required'] = []; - $schema['properties'] = []; - - foreach ($this->getProperties() as $name => $property) { - if ($property['required'] === true) { - $schema['required'][] = $name; - } - unset($property['title'], $property['required']); - - // Remove empty fields with array_filter(), and add it to the properties of the schema. - $schema['properties'][$name] = array_filter($property); - } - - return json_decode(json_encode($schema)); + $data = $this->jsonSerialize(); + unset($data['id'], $data['uuid'], $data['summary'], $data['archive'], $data['source'], + $data['updated'], $data['created']); + + $data['type'] = 'object'; + + // required on properties will break the validator, only have it set on object level + // array_filter is used so that empty string or 0 validation rules are removed, so we dont validate what we didnt set + if (isset($data['properties']) === true && empty($data['properties']) === false) { + foreach ($data['properties'] as $key => $property) { + $title = $property['title'] ?? $key; + $data['properties'][$title] = array_filter($property); + if (array_key_exists('required', $data['properties'][$key])) { + unset($data['properties'][$key]['required']); + } + } + } + + // Validator needs this specific $schema + $data['$schema'] = 'https://json-schema.org/draft/2020-12/schema'; + $data['$id'] = $urlGenerator->getAbsoluteURL($urlGenerator->linkToRoute('opencatalogi.publication_types.show', ['id' => $this->getUuid()])); + + // var_dump(json_decode(json_encode($data))); + return json_decode(json_encode($data)); } } diff --git a/lib/Service/ValidationService.php b/lib/Service/ValidationService.php index 649f369d..be1fca22 100644 --- a/lib/Service/ValidationService.php +++ b/lib/Service/ValidationService.php @@ -69,7 +69,7 @@ public function validatePublication(array $publication): array } } - $result = $validator->validate(data: (object) json_decode(json_encode($publication['data'])), schema: $publicationType->getSchema($this->urlGenerator)); + $result = $validator->validate(data: (object) json_decode(json_encode($publication['data'])), schema: $publicationType->getSchemaObject($this->urlGenerator)); $publication['validation'] = [ 'errors' => [], From 384de550873c7eea5019a3675f2db55f3b0bb1d8 Mon Sep 17 00:00:00 2001 From: Barry Brands Date: Thu, 21 Nov 2024 16:00:43 +0100 Subject: [PATCH 10/18] remove var_dump --- lib/Db/PublicationType.php | 1 - 1 file changed, 1 deletion(-) diff --git a/lib/Db/PublicationType.php b/lib/Db/PublicationType.php index 61795df3..cfc9885d 100644 --- a/lib/Db/PublicationType.php +++ b/lib/Db/PublicationType.php @@ -142,7 +142,6 @@ public function getSchemaObject(IURLGenerator $urlGenerator): object $data['$schema'] = 'https://json-schema.org/draft/2020-12/schema'; $data['$id'] = $urlGenerator->getAbsoluteURL($urlGenerator->linkToRoute('opencatalogi.publication_types.show', ['id' => $this->getUuid()])); - // var_dump(json_decode(json_encode($data))); return json_decode(json_encode($data)); } } From 145d1aee731c3d94e1a81a44282d7b92a0f085a5 Mon Sep 17 00:00:00 2001 From: Remko Date: Thu, 21 Nov 2024 16:46:57 +0100 Subject: [PATCH 11/18] added workflows --- .../pull-request-from-branch-check.yaml | 18 +++++++++++++++++ .../workflows/pull-request-lint-check.yaml | 20 +++++++++++++++++++ 2 files changed, 38 insertions(+) create mode 100644 .github/workflows/pull-request-from-branch-check.yaml create mode 100644 .github/workflows/pull-request-lint-check.yaml diff --git a/.github/workflows/pull-request-from-branch-check.yaml b/.github/workflows/pull-request-from-branch-check.yaml new file mode 100644 index 00000000..a40f09fa --- /dev/null +++ b/.github/workflows/pull-request-from-branch-check.yaml @@ -0,0 +1,18 @@ +name: Main Branch Protection + +on: + pull_request: + branches: + - master + +jobs: + check-branch: + runs-on: ubuntu-latest + steps: + - name: Check branch + run: | + if [[ ${GITHUB_HEAD_REF} != development ]] && [[ ${GITHUB_HEAD_REF} != documentation ]] && ! [[ ${GITHUB_HEAD_REF} =~ ^hotfix/ ]]; + then + echo "Error: Pull request must come from 'development', 'documentation' or 'hotfix/' branch" + exit 1 + fi diff --git a/.github/workflows/pull-request-lint-check.yaml b/.github/workflows/pull-request-lint-check.yaml new file mode 100644 index 00000000..0c04e87a --- /dev/null +++ b/.github/workflows/pull-request-lint-check.yaml @@ -0,0 +1,20 @@ +name: Lint Check + +on: + pull_request: + branches: + - never + +jobs: + lint-check: + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@v2 + + - name: Install dependencies + run: npm i + + - name: Linting + run: npm run lint From fc43d0ed4d14bd70cbdbb3c35cfab4d7b3685186 Mon Sep 17 00:00:00 2001 From: Ruben van der Linde Date: Fri, 29 Nov 2024 15:16:49 +0100 Subject: [PATCH 12/18] Add pages to open catalogi --- appinfo/routes.php | 5 +- lib/Controller/PagesController | 145 +++++++++++++ lib/Controller/SettingsController.php | 7 +- lib/Db/Page.php | 112 ++++++++++ lib/Db/PageMapper.php | 150 ++++++++++++++ lib/Migration/Version6Date20241129151236.php | 74 +++++++ lib/Service/ObjectService.php | 4 + src/entities/index.js | 1 + src/entities/page/index.js | 4 + src/entities/page/page.mock.ts | 43 ++++ src/entities/page/page.spec.ts | 47 +++++ src/entities/page/page.ts | 74 +++++++ src/entities/page/page.types.ts | 13 ++ src/navigation/MainMenu.vue | 7 + src/store/modules/page.spec.ts | 53 +++++ src/store/modules/page.ts | 160 +++++++++++++++ src/store/store.js | 3 + src/views/Views.vue | 3 + src/views/pages/PageDetail.vue | 202 +++++++++++++++++++ src/views/pages/PageIndex.vue | 52 +++++ src/views/pages/PageList.vue | 202 +++++++++++++++++++ 21 files changed, 1358 insertions(+), 3 deletions(-) create mode 100644 lib/Controller/PagesController create mode 100644 lib/Db/Page.php create mode 100644 lib/Db/PageMapper.php create mode 100644 lib/Migration/Version6Date20241129151236.php create mode 100644 src/entities/page/index.js create mode 100644 src/entities/page/page.mock.ts create mode 100644 src/entities/page/page.spec.ts create mode 100644 src/entities/page/page.ts create mode 100644 src/entities/page/page.types.ts create mode 100644 src/store/modules/page.spec.ts create mode 100644 src/store/modules/page.ts create mode 100644 src/views/pages/PageDetail.vue create mode 100644 src/views/pages/PageIndex.vue create mode 100644 src/views/pages/PageList.vue diff --git a/appinfo/routes.php b/appinfo/routes.php index 427b3af4..311f82ae 100644 --- a/appinfo/routes.php +++ b/appinfo/routes.php @@ -6,6 +6,7 @@ 'publications' => ['url' => '/api/publications'], 'organizations' => ['url' => '/api/organizations'], 'themes' => ['url' => '/api/themes'], + 'pages' => ['url' => '/api/pages'], 'attachments' => ['url' => '/api/attachments'], 'catalogi' => ['url' => '/api/catalogi'], 'listings' => ['url' => '/api/listings'], @@ -37,6 +38,8 @@ ['name' => 'search#publication', 'url' => '/api/search/publications/{publicationId}', 'verb' => 'GET', 'requirements' => ['publicationId' => '[^/]+']], ['name' => 'search#attachments', 'url' => '/api/search/publications/{publicationId}/attachments', 'verb' => 'GET', 'requirements' => ['publicationId' => '[^/]+']], ['name' => 'search#themes', 'url' => '/api/search/themes', 'verb' => 'GET'], - ['name' => 'search#theme', 'url' => '/api/search/themes/{themeId}', 'verb' => 'GET', 'requirements' => ['themeId' => '\d+']] + ['name' => 'search#theme', 'url' => '/api/search/themes/{themeId}', 'verb' => 'GET', 'requirements' => ['themeId' => '\d+']], + ['name' => 'search#pages', 'url' => '/api/search/pages', 'verb' => 'GET'], + ['name' => 'search#page', 'url' => '/api/search/pages/{pageSlug}', 'verb' => 'GET', 'requirements' => ['pageId' => '[^/]+']] ] ]; diff --git a/lib/Controller/PagesController b/lib/Controller/PagesController new file mode 100644 index 00000000..d10ef7d5 --- /dev/null +++ b/lib/Controller/PagesController @@ -0,0 +1,145 @@ +request->getParams(); + + // Fetch page objects based on filters and order + $data = $this->objectService->getResultArrayForRequest('page', $requestParams); + + // Return JSON response + return new JSONResponse($data); + } + + /** + * Retrieve a specific page by its ID. + * + * @param string|int $id The ID of the page to retrieve + * @return JSONResponse JSON response containing the requested page + * + * @NoAdminRequired + * @NoCSRFRequired + */ + public function show(string|int $id): JSONResponse + { + // Fetch the page object by its ID + $object = $this->objectService->getObject('page', $id); + + // Return the page as a JSON response + return new JSONResponse($object); + } + + /** + * Create a new page. + * + * @return JSONResponse The response containing the created page object. + * + * @NoAdminRequired + * @NoCSRFRequired + */ + public function create(): JSONResponse + { + // Get all parameters from the request + $data = $this->request->getParams(); + + // Remove the 'id' field if it exists, as we're creating a new object + unset($data['id']); + + // Save the new page object + $object = $this->objectService->saveObject('page', $data); + + // Return the created object as a JSON response + return new JSONResponse($object); + } + + /** + * Update an existing page. + * + * @param string|int $id The ID of the page to update. + * @return JSONResponse The response containing the updated page object. + * + * @NoAdminRequired + * @NoCSRFRequired + */ + public function update(string|int $id): JSONResponse + { + // Get all parameters from the request + $data = $this->request->getParams(); + + // Ensure the ID in the data matches the ID in the URL + $data['id'] = $id; + + // Save the updated page object + $object = $this->objectService->saveObject('page', $data); + + // Return the updated object as a JSON response + return new JSONResponse($object); + } + + /** + * Delete a page. + * + * @param string|int $id The ID of the page to delete. + * @return JSONResponse The response indicating the result of the deletion. + * + * @NoAdminRequired + * @NoCSRFRequired + */ + public function destroy(string|int $id): JSONResponse + { + // Delete the page object + $result = $this->objectService->deleteObject('page', $id); + + // Return the result as a JSON response + return new JSONResponse(['success' => $result], $result === true ? '200' : '404'); + } +} diff --git a/lib/Controller/SettingsController.php b/lib/Controller/SettingsController.php index 8da08328..5d975be3 100644 --- a/lib/Controller/SettingsController.php +++ b/lib/Controller/SettingsController.php @@ -45,7 +45,7 @@ public function index(): JSONResponse { // Initialize the data array $data = []; - $data['objectTypes'] = ['attachment', 'catalog', 'listing', 'publicationtype', 'organization', 'publication', 'theme']; + $data['objectTypes'] = ['attachment', 'catalog', 'listing', 'publicationtype', 'organization', 'publication', 'theme', 'page']; $data['openRegisters'] = false; $data['availableRegisters'] = []; @@ -78,7 +78,10 @@ public function index(): JSONResponse 'publication_register' => '', 'theme_source' => 'internal', 'theme_schema' => '', - 'theme_register' => '' + 'theme_register' => '', + 'page_source' => 'internal', + 'page_schema' => '', + 'page_register' => '', ]; // Get the current values for the object types from the configuration diff --git a/lib/Db/Page.php b/lib/Db/Page.php new file mode 100644 index 00000000..30f6020f --- /dev/null +++ b/lib/Db/Page.php @@ -0,0 +1,112 @@ +addType(fieldName: 'uuid', type: 'string'); + $this->addType(fieldName: 'name', type: 'string'); + $this->addType(fieldName: 'slug', type: 'string'); + $this->addType(fieldName: 'contents', type: 'json'); + $this->addType(fieldName: 'created', type: 'datetime'); + $this->addType(fieldName: 'updated', type: 'datetime'); + } + + /** + * Get array of JSON field names + * + * @return array List of field names that are JSON type + */ + public function getJsonFields(): array + { + return array_keys( + array_filter($this->getFieldTypes(), function ($field) { + return $field === 'json'; + }) + ); + } + + /** + * Hydrate the entity from an array of data + * + * @param array $object Data to hydrate from + * @return self + */ + public function hydrate(array $object): self + { + $jsonFields = $this->getJsonFields(); + + // Remove any fields that start with an underscore + // These are typically internal fields that shouldn't be updated directly + foreach ($object as $key => $value) { + if (str_starts_with($key, '_')) { + unset($object[$key]); + } + } + + foreach ($object as $key => $value) { + if (in_array($key, $jsonFields) === true && $value === []) { + $value = null; + } + + $method = 'set'.ucfirst($key); + + try { + $this->$method($value); + } catch (\Exception $exception) { + } + } + + return $this; + } + + /** + * Serialize the entity to JSON + * + * @return array Serialized data + */ + public function jsonSerialize(): array + { + $array = [ + 'id' => $this->id, + 'uuid' => $this->uuid, + 'name' => $this->name, + 'slug' => $this->slug, + 'contents' => $this->contents, + 'created_at' => $this->created?->format('c'), + 'updated_at' => $this->updated?->format('c') + ]; + + $jsonFields = $this->getJsonFields(); + + foreach ($array as $key => $value) { + if (in_array($key, $jsonFields) === true && $value === null) { + $array[$key] = []; + } + } + + return $array; + } +} diff --git a/lib/Db/PageMapper.php b/lib/Db/PageMapper.php new file mode 100644 index 00000000..03bcb026 --- /dev/null +++ b/lib/Db/PageMapper.php @@ -0,0 +1,150 @@ +db->getQueryBuilder(); + + $qb->select('*') + ->from('ocat_pages') + ->where($qb->expr()->orX( + $qb->expr()->eq('id', $qb->createNamedParameter($id, IQueryBuilder::PARAM_INT)), + $qb->expr()->eq('uuid', $qb->createNamedParameter($id, IQueryBuilder::PARAM_STR)) + )); + + try { + return $this->findEntity($qb); + } catch (\OCP\AppFramework\Db\DoesNotExistException $e) { + return null; + } + } + + /** + * Find multiple Pages by their IDs or UUIDs + * + * @param array $ids An array of IDs or UUIDs + * @return array An array of found Page entities + */ + public function findMultiple(array $ids): array + { + $qb = $this->db->getQueryBuilder(); + + $qb->select('*') + ->from('ocat_pages') + ->where($qb->expr()->orX( + $qb->expr()->in('id', $qb->createNamedParameter($ids, IQueryBuilder::PARAM_INT_ARRAY)), + $qb->expr()->in('uuid', $qb->createNamedParameter($ids, IQueryBuilder::PARAM_STR_ARRAY)) + )); + + return $this->findEntities(query: $qb); + } + + /** + * Find all Pages with optional limit and offset + * + * @param int|null $limit Maximum number of results to return + * @param int|null $offset Number of results to skip + * @return array An array of all found Page entities + */ + public function findAll(int $limit = null, int $offset = null, array $filters = [], array $sort = [], ?string $search = null): array + { + $qb = $this->db->getQueryBuilder(); + + $qb->select('*') + ->from('ocat_pages') + ->setMaxResults($limit) + ->setFirstResult($offset); + + return $this->findEntities(query: $qb); + } + + /** + * Create a new Page from an array of data + * + * @param array $object An array of Page data + * @return Page The newly created Page entity + */ + public function createFromArray(array $object): Page + { + $page = new Page(); + $page->hydrate(object: $object); + + // Set uuid if not provided + if ($page->getUuid() === null) { + $page->setUuid(Uuid::v4()); + } + + // Generate slug from name if not provided + if ($page->getSlug() === null && $page->getName() !== null) { + // Convert to lowercase and replace spaces with dashes + $slug = strtolower($page->getName()); + $slug = preg_replace('/[^a-z0-9-]/', '-', $slug); + $slug = preg_replace('/-+/', '-', $slug); + $slug = trim($slug, '-'); + $page->setSlug($slug); + } + + return $this->insert(entity: $page); + } + + /** + * Update an existing Page from an array of data + * + * @param int $id The ID of the Page to update + * @param array $object An array of updated Page data + * @return Page The updated Page entity + * @throws DoesNotExistException If the entity is not found + * @throws MultipleObjectsReturnedException|\OCP\DB\Exception If multiple entities are found + */ + public function updateFromArray(int $id, array $object): Page + { + $page = $this->find($id); + // Fallback to create if the page does not exist + if ($page === null) { + $object['uuid'] = $id; + return $this->createFromArray($object); + } + + $page->hydrate($object); + + return $this->update($page); + } +} diff --git a/lib/Migration/Version6Date20241129151236.php b/lib/Migration/Version6Date20241129151236.php new file mode 100644 index 00000000..16192cb8 --- /dev/null +++ b/lib/Migration/Version6Date20241129151236.php @@ -0,0 +1,74 @@ +hasTable(tableName: 'ocat_pages') === false) { + $table = $schema->createTable(tableName: 'ocat_pages'); + + // Primary key and identifier columns + $table->addColumn(name: 'id', typeName: Types::BIGINT, options: ['autoincrement' => true, 'notnull' => true, 'length' => 4]); + $table->addColumn(name: 'uuid', typeName: Types::STRING, options: ['notnull' => true, 'length' => 255]); + $table->addColumn(name: 'version', typeName: Types::STRING, options: ['notnull' => true, 'length' => 255, 'default' => '0.0.1']); + + // Meta columns + $table->addColumn(name: 'name', typeName: Types::STRING, options: ['notnull' => true, 'length' => 255]); + $table->addColumn(name: 'slug', typeName: Types::STRING, options: ['notnull' => true, 'length' => 255]); + $table->addColumn(name: 'contents', typeName: Types::JSON, options: ['notnull' => false]); + $table->addColumn(name: 'updated', typeName: Types::DATETIME, options: ['notnull' => true, 'default' => 'CURRENT_TIMESTAMP']); + $table->addColumn(name: 'created', typeName: Types::DATETIME, options: ['notnull' => true, 'default' => 'CURRENT_TIMESTAMP']); + + // Keys and indexes + $table->setPrimaryKey(columnNames: ['id']); + $table->addIndex(['uuid'], 'ocat_pages_uuid_index'); + $table->addIndex(['slug'], 'ocat_pages_slug_index'); + } + + return $schema; + } + + /** + * @param IOutput $output + * @param Closure(): ISchemaWrapper $schemaClosure + * @param array $options + */ + public function postSchemaChange(IOutput $output, Closure $schemaClosure, array $options): void { + } +} diff --git a/lib/Service/ObjectService.php b/lib/Service/ObjectService.php index e0befbbf..e223d22f 100644 --- a/lib/Service/ObjectService.php +++ b/lib/Service/ObjectService.php @@ -20,6 +20,7 @@ use OCP\IAppConfig; // Import mappers use OCA\OpenCatalogi\Db\AttachmentMapper; +use OCA\OpenCatalogi\Db\PageMapper; use OCA\OpenCatalogi\Db\CatalogMapper; use OCA\OpenCatalogi\Db\ListingMapper; use OCA\OpenCatalogi\Db\PublicationTypeMapper; @@ -47,6 +48,7 @@ class ObjectService * @param OrganizationMapper $organizationMapper Mapper for organizations * @param PublicationMapper $publicationMapper Mapper for publications * @param ThemeMapper $themeMapper Mapper for themes + * @param PageMapper $pageMapper Mapper for pages * @param ContainerInterface $container Container for dependency injection * @param IAppManager $appManager App manager interface * @param IAppConfig $config App configuration interface @@ -59,6 +61,7 @@ public function __construct( private OrganizationMapper $organizationMapper, private PublicationMapper $publicationMapper, private ThemeMapper $themeMapper, + private PageMapper $pageMapper, private ContainerInterface $container, private readonly IAppManager $appManager, private readonly IAppConfig $config, @@ -112,6 +115,7 @@ private function getMapper(string $objectType): mixed 'organization' => $this->organizationMapper, 'publication' => $this->publicationMapper, 'theme' => $this->themeMapper, + 'page' => $this->pageMapper, default => throw new InvalidArgumentException("Unknown object type: $objectType"), }; } diff --git a/src/entities/index.js b/src/entities/index.js index 2f4bb056..b5726d33 100644 --- a/src/entities/index.js +++ b/src/entities/index.js @@ -7,3 +7,4 @@ export * from './organization/index.js' export * from './publication/index.js' export * from './theme/index.js' export * from './publicationType/index.js' +export * from './page/index.js' \ No newline at end of file diff --git a/src/entities/page/index.js b/src/entities/page/index.js new file mode 100644 index 00000000..b496e408 --- /dev/null +++ b/src/entities/page/index.js @@ -0,0 +1,4 @@ +export * from './page.ts' +export * from './page.types.ts' +export * from './page.mock.ts' + diff --git a/src/entities/page/page.mock.ts b/src/entities/page/page.mock.ts new file mode 100644 index 00000000..90e5e8f5 --- /dev/null +++ b/src/entities/page/page.mock.ts @@ -0,0 +1,43 @@ +import { Page } from './page' +import { TPage } from './page.types' + +/** + * Mock data function that returns an array of page data objects + * Used for testing and development purposes + */ +export const mockPageData = (): TPage[] => [ + { // full data + id: '1', + uuid: '123e4567-e89b-12d3-a456-426614174000', + name: 'Test Page', + slug: 'test-page', + contents: [], + createdAt: new Date().toISOString(), + updatedAt: new Date().toISOString() + }, + // @ts-expect-error -- expected missing contents + { // partial data + id: '2', + uuid: '123e4567-e89b-12d3-a456-426614174001', + name: 'Another Page', + slug: 'another-page', + createdAt: new Date().toISOString(), + updatedAt: new Date().toISOString() + }, + { // invalid data + id: '3', + uuid: '123e4567-e89b-12d3-a456-426614174002', + name: '', + slug: '', + contents: [], + createdAt: new Date().toISOString(), + updatedAt: new Date().toISOString() + }, +] + +/** + * Creates an array of Page instances from provided data or default mock data + * @param data Optional array of page data to convert to Page instances + * @returns Array of Page instances + */ +export const mockPage = (data: TPage[] = mockPageData()): TPage[] => data.map(item => new Page(item)) diff --git a/src/entities/page/page.spec.ts b/src/entities/page/page.spec.ts new file mode 100644 index 00000000..fa342192 --- /dev/null +++ b/src/entities/page/page.spec.ts @@ -0,0 +1,47 @@ +/* eslint-disable no-console */ +import { Page } from './page' +import { mockPage } from './page.mock' + +describe('Page Store', () => { + it('create Page entity with full data', () => { + const page = new Page(mockPage()[0]) + + expect(page).toBeInstanceOf(Page) + expect(page).toEqual(mockPage()[0]) + expect(page.uuid).toBe(mockPage()[0].uuid) + expect(page.name).toBe(mockPage()[0].name) + expect(page.slug).toBe(mockPage()[0].slug) + expect(page.contents).toEqual(mockPage()[0].contents) + expect(page.createdAt).toBe(mockPage()[0].createdAt) + expect(page.updatedAt).toBe(mockPage()[0].updatedAt) + + expect(page.validate().success).toBe(true) + }) + + it('create Page entity with partial data', () => { + const page = new Page(mockPage()[1]) + + expect(page).toBeInstanceOf(Page) + expect(page.id).toBe(mockPage()[1].id) + expect(page.uuid).toBe(mockPage()[1].uuid) + expect(page.name).toBe(mockPage()[1].name) + expect(page.slug).toBe(mockPage()[1].slug) + expect(page.contents).toBeNull() + expect(page.createdAt).toBe(mockPage()[1].createdAt) + expect(page.updatedAt).toBe(mockPage()[1].updatedAt) + + expect(page.validate().success).toBe(true) + }) + + it('create Page entity with falsy data', () => { + const page = new Page(mockPage()[2]) + + expect(page).toBeInstanceOf(Page) + expect(page).toEqual(mockPage()[2]) + expect(page.uuid).toBe(mockPage()[2].uuid) + expect(page.name).toBe('') + expect(page.contents).toBeNull() + + expect(page.validate().success).toBe(false) + }) +}) diff --git a/src/entities/page/page.ts b/src/entities/page/page.ts new file mode 100644 index 00000000..978eaf07 --- /dev/null +++ b/src/entities/page/page.ts @@ -0,0 +1,74 @@ +import { SafeParseReturnType, z } from 'zod' +import { TPage } from './page.types' + +/** + * Page class representing a page entity with validation + * Implements the TPage interface for type safety + */ +export class Page implements TPage { + + public id: string + public uuid: string + public name: string + public slug: string + public contents: Array + public createdAt: string + public updatedAt: string + + /** + * Creates a new Page instance + * @param data Initial page data conforming to TPage interface + */ + constructor(data: TPage) { + this.hydrate(data) + } + + /* istanbul ignore next */ // Jest does not recognize the code coverage of these 2 methods + /** + * Hydrates the page object with provided data + * @param data Page data to populate the instance + */ + private hydrate(data: TPage) { + this.id = data?.id?.toString() || '' + this.uuid = data?.uuid || '' + this.name = data?.name || '' + this.slug = data?.slug || '' + this.contents = data?.contents?.map(contents => ({ + title: contents?.title || '', + summary: contents?.summary || '', + description: contents?.description || '', + image: contents?.image || '' + })) || [] + this.createdAt = data?.createdAt || '' + this.updatedAt = data?.updatedAt || '' + } + + /* istanbul ignore next */ + /** + * Validates the page data against a schema + * @returns SafeParseReturnType containing validation result + */ + public validate(): SafeParseReturnType { + // Schema validation for page data + const schema = z.object({ + uuid: z.string().min(1, 'is verplicht'), + name: z.string().min(1, 'is verplicht'), + slug: z.string().min(1, 'is verplicht'), + contents: z.array(z.object({ + title: z.string().min(1, 'is verplicht'), + summary: z.string().min(1, 'is verplicht'), + description: z.string(), + image: z.string() + })), + createdAt: z.string(), + updatedAt: z.string() + }) + + const result = schema.safeParse({ + ...this, + }) + + return result + } + +} diff --git a/src/entities/page/page.types.ts b/src/entities/page/page.types.ts new file mode 100644 index 00000000..2aea7d70 --- /dev/null +++ b/src/entities/page/page.types.ts @@ -0,0 +1,13 @@ +/** + * Type definition for a Page object + * Represents the structure of a page with content and metadata + */ +export type TPage = { + id: string // Unique identifier for the page + uuid: string // Unique identifier for the page + name: string // Title/heading of the page + contents: Array // Main content/body of the page - can contain any type of content + slug: string // URL-friendly version of the title + createdAt: string // Creation timestamp + updatedAt: string // Last update timestamp +} diff --git a/src/navigation/MainMenu.vue b/src/navigation/MainMenu.vue index 0b590dcf..896e5742 100644 --- a/src/navigation/MainMenu.vue +++ b/src/navigation/MainMenu.vue @@ -53,6 +53,11 @@ import { navigationStore, catalogiStore, publicationStore } from '../store/store + + + @@ -92,6 +97,7 @@ import Finance from 'vue-material-design-icons/Finance.vue' import BookOpenVariantOutline from 'vue-material-design-icons/BookOpenVariantOutline.vue' import OfficeBuildingOutline from 'vue-material-design-icons/OfficeBuildingOutline.vue' import ShapeOutline from 'vue-material-design-icons/ShapeOutline.vue' +import Web from 'vue-material-design-icons/Web.vue' export default { name: 'MainMenu', @@ -114,6 +120,7 @@ export default { BookOpenVariantOutline, OfficeBuildingOutline, ShapeOutline, + Web, }, data() { return { diff --git a/src/store/modules/page.spec.ts b/src/store/modules/page.spec.ts new file mode 100644 index 00000000..be1dbb7a --- /dev/null +++ b/src/store/modules/page.spec.ts @@ -0,0 +1,53 @@ +/* eslint-disable no-console */ +import { setActivePinia, createPinia } from 'pinia' + +import { usePageStore } from './page.js' +import { mockPage, Page } from '../../entities/index.js' + +describe('Page Store', () => { + beforeEach(() => { + setActivePinia(createPinia()) + }) + + it('sets page item correctly', () => { + const store = usePageStore() + + store.setPageItem(mockPage()[0]) + + expect(store.pageItem).toBeInstanceOf(Page) + expect(store.pageItem).toEqual(mockPage()[0]) + expect(store.pageItem.validate().success).toBe(true) + + store.setPageItem(mockPage()[1]) + + expect(store.pageItem).toBeInstanceOf(Page) + expect(store.pageItem).toEqual(mockPage()[1]) + expect(store.pageItem.validate().success).toBe(true) + + store.setPageItem(mockPage()[2]) + + expect(store.pageItem).toBeInstanceOf(Page) + expect(store.pageItem).toEqual(mockPage()[2]) + expect(store.pageItem.validate().success).toBe(false) + }) + + it('sets page list correctly', () => { + const store = usePageStore() + + store.setPageList(mockPage()) + + expect(store.pageList).toHaveLength(mockPage().length) + + expect(store.pageList[0]).toBeInstanceOf(Page) + expect(store.pageList[0]).toEqual(mockPage()[0]) + expect(store.pageList[0].validate().success).toBe(true) + + expect(store.pageList[1]).toBeInstanceOf(Page) + expect(store.pageList[1]).toEqual(mockPage()[1]) + expect(store.pageList[1].validate().success).toBe(true) + + expect(store.pageList[2]).toBeInstanceOf(Page) + expect(store.pageList[2]).toEqual(mockPage()[2]) + expect(store.pageList[2].validate().success).toBe(false) + }) +}) diff --git a/src/store/modules/page.ts b/src/store/modules/page.ts new file mode 100644 index 00000000..0d132a7c --- /dev/null +++ b/src/store/modules/page.ts @@ -0,0 +1,160 @@ +/* eslint-disable no-console */ +import { Page, TPage } from '../../entities/index.js' +import { defineStore } from 'pinia' + +const apiEndpoint = '/index.php/apps/opencatalogi/api/pages' + +interface Options { + /** + * Do not save the received item to the store, this can be enabled if API calls get run in a loop + */ + doNotSetStore?: boolean +} + +interface PageStoreState { + pageItem: Page; + pageList: Page[]; +} + +export const usePageStore = defineStore('page', { + state: () => ({ + pageItem: null, + pageList: [], + } as PageStoreState), + actions: { + setPageItem(pageItem: Page | TPage) { + this.pageItem = pageItem && new Page(pageItem as TPage) + console.log('Active page item set to ' + pageItem && pageItem?.id) + }, + setPageList(pageList: Page[] | TPage[]) { + this.pageList = pageList.map( + (pageItem) => new Page(pageItem as TPage), + ) + console.log('Page list set to ' + pageList.length + ' items') + }, + /* istanbul ignore next */ // ignore this for Jest until moved into a service + async refreshPageList(search: string = null) { + // @todo this might belong in a service? + let endpoint = apiEndpoint + if (search !== null && search !== '') { + endpoint = endpoint + '?_search=' + search + } + const response = await fetch( + endpoint, { + method: 'GET', + }, + ) + const rawData = (await response.json()).results + const data = rawData.map((pageItem: TPage) => new Page(pageItem)) + + this.setPageList(data) + + return { response, data } + }, + /* istanbul ignore next */ + async getAllPages(options: Options = {}) { + const response = await fetch( + `${apiEndpoint}`, + { method: 'get' }, + ) + + const rawData = await response.json() + + const data = rawData.results.map((pageItem: TPage) => new Page(pageItem)) + + options.doNotSetStore !== true && this.setPageList(data) + + return { response, data } + }, + /* istanbul ignore next */ + async getOnePage(id: number, options: Options = {}) { + if (!id) { + throw Error('Passed id is falsy') + } + + const response = await fetch( + `${apiEndpoint}/${id}`, + { method: 'get' }, + ) + + const data = new Page(await response.json()) + + options.doNotSetStore !== true && this.setPageItem(data) + + return { response, data } + }, + /* istanbul ignore next */ + async addPage(item: Page) { + if (!(item instanceof Page)) { + throw Error('Please pass a Page item from the Page class') + } + + const validateResult = item.validate() + if (!validateResult.success) { + throw Error(validateResult.error.issues[0].message) + } + + const response = await fetch(apiEndpoint, + { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify(validateResult.data), + }, + ) + + const data = new Page(await response.json()) + + this.refreshPageList() + this.setPageItem(data) + + return { response, data } + }, + /* istanbul ignore next */ + async editPage(pageItem: Page) { + if (!(pageItem instanceof Page)) { + throw Error('Please pass a Page item from the Page class') + } + + const validateResult = pageItem.validate() + if (!validateResult.success) { + throw Error(validateResult.error.issues[0].message) + } + + const response = await fetch( + `${apiEndpoint}/${pageItem.id}`, + { + method: 'PUT', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify(validateResult.data), + }, + ) + + const data = new Page(await response.json()) + + this.refreshPageList() + this.setPageItem(data) + + return { response, data } + }, + /* istanbul ignore next */ + async deletePage(id: number) { + if (!id) { + throw Error('Passed id is falsy') + } + + const response = await fetch( + `${apiEndpoint}/${id}`, + { method: 'DELETE' }, + ) + + this.refreshPageList() + this.setPageItem(null) + + return { response } + }, + }, +}) diff --git a/src/store/store.js b/src/store/store.js index 1eddc424..0f1b9e7d 100644 --- a/src/store/store.js +++ b/src/store/store.js @@ -14,6 +14,7 @@ import { useOrganizationStore } from './modules/organization' import { usePublicationStore } from './modules/publication' import { useSearchStore } from './modules/search' import { useThemeStore } from './modules/theme' +import { usePageStore } from './modules/page' const navigationStore = useNavigationStore(pinia) const searchStore = useSearchStore(pinia) @@ -24,6 +25,7 @@ const publicationStore = usePublicationStore(pinia) const organizationStore = useOrganizationStore(pinia) const themeStore = useThemeStore(pinia) const configurationStore = useConfigurationStore(pinia) +const pageStore = usePageStore(pinia) export { // generic @@ -37,4 +39,5 @@ export { organizationStore, themeStore, configurationStore, + pageStore, } diff --git a/src/views/Views.vue b/src/views/Views.vue index 8ba8cae6..ca810743 100644 --- a/src/views/Views.vue +++ b/src/views/Views.vue @@ -9,6 +9,7 @@ import { navigationStore } from '../store/store.js' + @@ -23,6 +24,7 @@ import { NcAppContent } from '@nextcloud/vue' import Catalogi from './catalogi/CatalogiIndex.vue' import Organizations from './organizations/OrganizationIndex.vue' import Themes from './themes/ThemeIndex.vue' +import Pages from './pages/PageIndex.vue' import Dashboard from './dashboard/DashboardIndex.vue' import Directory from './directory/DirectoryIndex.vue' import PublicationType from './publicationType/PublicationTypeIndex.vue' @@ -35,6 +37,7 @@ export default { Catalogi, Organizations, Themes, + Pages, Dashboard, Directory, PublicationType, diff --git a/src/views/pages/PageDetail.vue b/src/views/pages/PageDetail.vue new file mode 100644 index 00000000..e01d6930 --- /dev/null +++ b/src/views/pages/PageDetail.vue @@ -0,0 +1,202 @@ + + + + + + + diff --git a/src/views/pages/PageIndex.vue b/src/views/pages/PageIndex.vue new file mode 100644 index 00000000..8d1f20cf --- /dev/null +++ b/src/views/pages/PageIndex.vue @@ -0,0 +1,52 @@ + + + + + diff --git a/src/views/pages/PageList.vue b/src/views/pages/PageList.vue new file mode 100644 index 00000000..49115cfe --- /dev/null +++ b/src/views/pages/PageList.vue @@ -0,0 +1,202 @@ + + + + + From 9fa9f7125ec4314bc51accdce53f04fc2ca7e05a Mon Sep 17 00:00:00 2001 From: Ruben van der Linde Date: Fri, 29 Nov 2024 15:26:23 +0100 Subject: [PATCH 13/18] First fixes --- appinfo/info.xml | 2 +- lib/Controller/{PagesController => PagesController.php} | 0 src/navigation/MainMenu.vue | 4 ++-- src/views/pages/PageDetail.vue | 2 ++ src/views/pages/PageIndex.vue | 6 +++--- src/views/pages/PageList.vue | 6 +++--- 6 files changed, 11 insertions(+), 9 deletions(-) rename lib/Controller/{PagesController => PagesController.php} (100%) diff --git a/appinfo/info.xml b/appinfo/info.xml index eb985017..e9496ccd 100644 --- a/appinfo/info.xml +++ b/appinfo/info.xml @@ -22,7 +22,7 @@ Create a [bug report](https://github.com/OpenCatalogi/.github/issues/new/choose) Create a [feature request](https://github.com/OpenCatalogi/.github/issues/new/choose) ]]> - 0.6.42 + 0.6.43 agpl Conduction Acato diff --git a/lib/Controller/PagesController b/lib/Controller/PagesController.php similarity index 100% rename from lib/Controller/PagesController rename to lib/Controller/PagesController.php diff --git a/src/navigation/MainMenu.vue b/src/navigation/MainMenu.vue index 896e5742..cb984d36 100644 --- a/src/navigation/MainMenu.vue +++ b/src/navigation/MainMenu.vue @@ -54,12 +54,12 @@ import { navigationStore, catalogiStore, publicationStore } from '../store/store diff --git a/src/views/pages/PageDetail.vue b/src/views/pages/PageDetail.vue index e01d6930..031a44b7 100644 --- a/src/views/pages/PageDetail.vue +++ b/src/views/pages/PageDetail.vue @@ -80,6 +80,7 @@ import Delete from 'vue-material-design-icons/Delete.vue' import DotsHorizontal from 'vue-material-design-icons/DotsHorizontal.vue' import HelpCircleOutline from 'vue-material-design-icons/HelpCircleOutline.vue' import Pencil from 'vue-material-design-icons/Pencil.vue' +import Web from 'vue-material-design-icons/Web.vue' /** * Component for displaying and managing page details @@ -97,6 +98,7 @@ export default { Delete, ContentCopy, HelpCircleOutline, + Web, }, props: { pageItem: { diff --git a/src/views/pages/PageIndex.vue b/src/views/pages/PageIndex.vue index 8d1f20cf..5e4cbbb3 100644 --- a/src/views/pages/PageIndex.vue +++ b/src/views/pages/PageIndex.vue @@ -13,7 +13,7 @@ import { navigationStore, searchStore, pageStore } from '../../store/store.js' name="Geen pagina" description="Nog geen pagina geselecteerd">