From ded60d16bf9fe4ab15f2d18d8aeeede6bc0cff4f Mon Sep 17 00:00:00 2001 From: Brandon Shipley <bshipley@hipowered.dev> Date: Sun, 24 Nov 2024 18:38:29 -0800 Subject: [PATCH] bring into standalone plugin for distribution --- .gitignore | 8 + README.md | 11 +- composer.json | 24 + .../20241114080900_CreateProductCatalogs.php | 44 ++ ...20241114081036_CreateProductCategories.php | 78 +++ .../20241114090547_CreateProducts.php | 47 ++ ...054627_CreateProductCategoryAttributes.php | 55 ++ ..._CreateProductCategoryAttributeOptions.php | 48 ++ ...22070040_CreateExternalProductCatalogs.php | 52 ++ config/app.example.php | 28 + phpunit.xml.dist | 30 ++ src/CakeProductsPlugin.php | 95 ++++ src/Controller/AppController.php | 10 + .../ExternalProductCatalogsController.php | 110 ++++ src/Controller/ProductCatalogsController.php | 111 ++++ .../ProductCategoriesController.php | 115 ++++ ...ductCategoryAttributeOptionsController.php | 48 ++ .../ProductCategoryAttributesController.php | 144 +++++ src/Controller/ProductsController.php | 110 ++++ src/Model/Entity/ExternalProductCatalog.php | 41 ++ src/Model/Entity/Product.php | 35 ++ src/Model/Entity/ProductCatalog.php | 37 ++ src/Model/Entity/ProductCategory.php | 49 ++ src/Model/Entity/ProductCategoryAttribute.php | 39 ++ .../Entity/ProductCategoryAttributeOption.php | 37 ++ .../Enum/ProductCategoryAttributeTypeId.php | 23 + src/Model/Enum/ProductProductTypeId.php | 23 + .../Table/ExternalProductCatalogsTable.php | 115 ++++ src/Model/Table/ProductCatalogsTable.php | 99 ++++ src/Model/Table/ProductCategoriesTable.php | 164 ++++++ .../ProductCategoryAttributeOptionsTable.php | 103 ++++ .../Table/ProductCategoryAttributesTable.php | 113 ++++ src/Model/Table/ProductsTable.php | 105 ++++ src/Service/CatalogManagerServiceProvider.php | 29 + src/Service/ExternalCatalogManagerService.php | 184 +++++++ src/Service/InternalCatalogManagerService.php | 106 ++++ templates/ExternalProductCatalogs/add.php | 32 ++ templates/ExternalProductCatalogs/edit.php | 37 ++ templates/ExternalProductCatalogs/index.php | 54 ++ templates/ExternalProductCatalogs/view.php | 52 ++ templates/ProductCatalogs/add.php | 29 + templates/ProductCatalogs/edit.php | 34 ++ templates/ProductCatalogs/index.php | 48 ++ templates/ProductCatalogs/view.php | 73 +++ templates/ProductCategories/add.php | 33 ++ templates/ProductCategories/edit.php | 38 ++ templates/ProductCategories/index.php | 50 ++ templates/ProductCategories/view.php | 95 ++++ .../ProductCategoryAttributeOptions/add.php | 20 + templates/ProductCategoryAttributes/add.php | 27 + templates/ProductCategoryAttributes/edit.php | 32 ++ templates/ProductCategoryAttributes/index.php | 50 ++ templates/ProductCategoryAttributes/view.php | 71 +++ templates/Products/add.php | 30 ++ templates/Products/edit.php | 35 ++ templates/Products/index.php | 48 ++ templates/Products/view.php | 40 ++ templates/element/Layout/submenu.php | 55 ++ .../ProductCategoryAttributes/form.php | 49 ++ ...product_category_attribute_option_form.php | 32 ++ .../ExternalProductCatalogsFixture.php | 33 ++ tests/Fixture/ProductCatalogsFixture.php | 36 ++ tests/Fixture/ProductCategoriesFixture.php | 101 ++++ ...ProductCategoryAttributeOptionsFixture.php | 31 ++ .../ProductCategoryAttributesFixture.php | 31 ++ tests/Fixture/ProductsFixture.php | 30 ++ .../Controller/BaseControllerTest.php | 25 + .../ExternalProductCatalogsControllerTest.php | 446 ++++++++++++++++ .../ProductCatalogsControllerTest.php | 450 ++++++++++++++++ .../ProductCategoriesControllerTest.php | 455 ++++++++++++++++ ...CategoryAttributeOptionsControllerTest.php | 198 +++++++ ...roductCategoryAttributesControllerTest.php | 498 ++++++++++++++++++ .../Controller/ProductsControllerTest.php | 452 ++++++++++++++++ .../ExternalProductCatalogsTableTest.php | 107 ++++ .../Model/Table/ProductCatalogsTableTest.php | 96 ++++ .../Table/ProductCategoriesTableTest.php | 111 ++++ ...oductCategoryAttributeOptionsTableTest.php | 104 ++++ .../ProductCategoryAttributesTableTest.php | 107 ++++ .../Model/Table/ProductsTableTest.php | 105 ++++ tests/bootstrap.php | 55 ++ tests/schema.sql | 1 + webroot/.gitkeep | 0 .../js/product_category_attribute_options.js | 15 + 83 files changed, 7020 insertions(+), 1 deletion(-) create mode 100644 .gitignore create mode 100644 composer.json create mode 100644 config/Migrations/20241114080900_CreateProductCatalogs.php create mode 100644 config/Migrations/20241114081036_CreateProductCategories.php create mode 100644 config/Migrations/20241114090547_CreateProducts.php create mode 100644 config/Migrations/20241115054627_CreateProductCategoryAttributes.php create mode 100644 config/Migrations/20241115062613_CreateProductCategoryAttributeOptions.php create mode 100644 config/Migrations/20241122070040_CreateExternalProductCatalogs.php create mode 100644 config/app.example.php create mode 100644 phpunit.xml.dist create mode 100644 src/CakeProductsPlugin.php create mode 100644 src/Controller/AppController.php create mode 100644 src/Controller/ExternalProductCatalogsController.php create mode 100644 src/Controller/ProductCatalogsController.php create mode 100644 src/Controller/ProductCategoriesController.php create mode 100644 src/Controller/ProductCategoryAttributeOptionsController.php create mode 100644 src/Controller/ProductCategoryAttributesController.php create mode 100644 src/Controller/ProductsController.php create mode 100644 src/Model/Entity/ExternalProductCatalog.php create mode 100644 src/Model/Entity/Product.php create mode 100644 src/Model/Entity/ProductCatalog.php create mode 100644 src/Model/Entity/ProductCategory.php create mode 100644 src/Model/Entity/ProductCategoryAttribute.php create mode 100644 src/Model/Entity/ProductCategoryAttributeOption.php create mode 100644 src/Model/Enum/ProductCategoryAttributeTypeId.php create mode 100644 src/Model/Enum/ProductProductTypeId.php create mode 100644 src/Model/Table/ExternalProductCatalogsTable.php create mode 100644 src/Model/Table/ProductCatalogsTable.php create mode 100644 src/Model/Table/ProductCategoriesTable.php create mode 100644 src/Model/Table/ProductCategoryAttributeOptionsTable.php create mode 100644 src/Model/Table/ProductCategoryAttributesTable.php create mode 100644 src/Model/Table/ProductsTable.php create mode 100644 src/Service/CatalogManagerServiceProvider.php create mode 100644 src/Service/ExternalCatalogManagerService.php create mode 100644 src/Service/InternalCatalogManagerService.php create mode 100644 templates/ExternalProductCatalogs/add.php create mode 100644 templates/ExternalProductCatalogs/edit.php create mode 100644 templates/ExternalProductCatalogs/index.php create mode 100644 templates/ExternalProductCatalogs/view.php create mode 100644 templates/ProductCatalogs/add.php create mode 100644 templates/ProductCatalogs/edit.php create mode 100644 templates/ProductCatalogs/index.php create mode 100644 templates/ProductCatalogs/view.php create mode 100644 templates/ProductCategories/add.php create mode 100644 templates/ProductCategories/edit.php create mode 100644 templates/ProductCategories/index.php create mode 100644 templates/ProductCategories/view.php create mode 100644 templates/ProductCategoryAttributeOptions/add.php create mode 100644 templates/ProductCategoryAttributes/add.php create mode 100644 templates/ProductCategoryAttributes/edit.php create mode 100644 templates/ProductCategoryAttributes/index.php create mode 100644 templates/ProductCategoryAttributes/view.php create mode 100644 templates/Products/add.php create mode 100644 templates/Products/edit.php create mode 100644 templates/Products/index.php create mode 100644 templates/Products/view.php create mode 100644 templates/element/Layout/submenu.php create mode 100644 templates/element/ProductCategoryAttributes/form.php create mode 100644 templates/element/ProductCategoryAttributes/product_category_attribute_option_form.php create mode 100644 tests/Fixture/ExternalProductCatalogsFixture.php create mode 100644 tests/Fixture/ProductCatalogsFixture.php create mode 100644 tests/Fixture/ProductCategoriesFixture.php create mode 100644 tests/Fixture/ProductCategoryAttributeOptionsFixture.php create mode 100644 tests/Fixture/ProductCategoryAttributesFixture.php create mode 100644 tests/Fixture/ProductsFixture.php create mode 100644 tests/TestCase/Controller/BaseControllerTest.php create mode 100644 tests/TestCase/Controller/ExternalProductCatalogsControllerTest.php create mode 100644 tests/TestCase/Controller/ProductCatalogsControllerTest.php create mode 100644 tests/TestCase/Controller/ProductCategoriesControllerTest.php create mode 100644 tests/TestCase/Controller/ProductCategoryAttributeOptionsControllerTest.php create mode 100644 tests/TestCase/Controller/ProductCategoryAttributesControllerTest.php create mode 100644 tests/TestCase/Controller/ProductsControllerTest.php create mode 100644 tests/TestCase/Model/Table/ExternalProductCatalogsTableTest.php create mode 100644 tests/TestCase/Model/Table/ProductCatalogsTableTest.php create mode 100644 tests/TestCase/Model/Table/ProductCategoriesTableTest.php create mode 100644 tests/TestCase/Model/Table/ProductCategoryAttributeOptionsTableTest.php create mode 100644 tests/TestCase/Model/Table/ProductCategoryAttributesTableTest.php create mode 100644 tests/TestCase/Model/Table/ProductsTableTest.php create mode 100644 tests/bootstrap.php create mode 100644 tests/schema.sql create mode 100644 webroot/.gitkeep create mode 100644 webroot/js/product_category_attribute_options.js diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..244d127 --- /dev/null +++ b/.gitignore @@ -0,0 +1,8 @@ +/composer.lock +/composer.phar +/phpunit.xml +/.phpunit.result.cache +/phpunit.phar +/config/Migrations/schema-dump-default.lock +/vendor/ +/.idea/ diff --git a/README.md b/README.md index 5b9deb6..615248c 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,11 @@ -# CakeProducts +# CakeProducts plugin for CakePHP +## Installation + +You can install this plugin into your CakePHP application using [composer](https://getcomposer.org). + +The recommended way to install composer packages is: + +``` +composer require your-name-here/cake-products +``` diff --git a/composer.json b/composer.json new file mode 100644 index 0000000..a3a7c2d --- /dev/null +++ b/composer.json @@ -0,0 +1,24 @@ +{ + "name": "your-name-here/cake-products", + "description": "CakeProducts plugin for CakePHP", + "type": "cakephp-plugin", + "license": "MIT", + "require": { + "php": ">=8.1", + "cakephp/cakephp": "^5.0.1" + }, + "require-dev": { + "phpunit/phpunit": "^10.1" + }, + "autoload": { + "psr-4": { + "CakeProducts\\": "src/" + } + }, + "autoload-dev": { + "psr-4": { + "CakeProducts\\Test\\": "tests/", + "Cake\\Test\\": "vendor/cakephp/cakephp/tests/" + } + } +} diff --git a/config/Migrations/20241114080900_CreateProductCatalogs.php b/config/Migrations/20241114080900_CreateProductCatalogs.php new file mode 100644 index 0000000..4140232 --- /dev/null +++ b/config/Migrations/20241114080900_CreateProductCatalogs.php @@ -0,0 +1,44 @@ +<?php +declare(strict_types=1); + +use Migrations\AbstractMigration; + +class CreateProductCatalogs extends AbstractMigration +{ + /** + * Change Method. + * + * More information on this method is available here: + * https://book.cakephp.org/phinx/0/en/migrations.html#the-change-method + * @return void + */ + public function change(): void + { + $table = $this->table('product_catalogs', ['id' => false, 'primary_key' => ['id']]); + $table->addColumn('id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('name', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('catalog_description', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => true, + ]); + $table->addColumn('enabled', 'boolean', [ + 'default' => null, + 'null' => false, + ]); + $table->addIndex([ + 'name', + ], [ + 'name' => 'BY_NAME', + 'unique' => true, + ]); + $table->create(); + } +} diff --git a/config/Migrations/20241114081036_CreateProductCategories.php b/config/Migrations/20241114081036_CreateProductCategories.php new file mode 100644 index 0000000..10631b0 --- /dev/null +++ b/config/Migrations/20241114081036_CreateProductCategories.php @@ -0,0 +1,78 @@ +<?php +declare(strict_types=1); + +use Migrations\AbstractMigration; + +class CreateProductCategories extends AbstractMigration +{ + /** + * Change Method. + * + * More information on this method is available here: + * https://book.cakephp.org/phinx/0/en/migrations.html#the-change-method + * @return void + */ + public function change(): void + { + $table = $this->table('product_categories'); + + $table->addColumn('product_catalog_id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('internal_id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('name', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('category_description', 'text', [ + 'default' => null, + 'null' => true, + ]); +// $table->addColumn('shopify_v1_id', 'integer', [ +// 'default' => null, +// 'limit' => 11, +// 'null' => true, +// ]); +// $table->addColumn('shopify_v2_id', 'string', [ +// 'default' => null, +// 'limit' => 255, +// 'null' => true, +// ]); + $table->addColumn('parent_id', 'integer', [ + 'default' => null, + 'limit' => 11, + 'null' => true, + ]); + $table->addColumn('lft', 'integer', [ + 'default' => null, + 'limit' => 11, + 'null' => false, + ]); + $table->addColumn('rght', 'integer', [ + 'default' => null, + 'limit' => 11, + 'null' => false, + ]); + $table->addColumn('enabled', 'boolean', [ + 'default' => false, + 'null' => false, + ]); + + $table->addIndex('parent_id'); + $table->addIndex('lft'); + $table->addIndex('product_catalog_id'); + $table->addIndex([ + 'product_catalog_id', + 'name', + ], [ + 'name' => 'BY_NAME_AND_CATALOG_ID', + 'unique' => true, + ]); + $table->create(); + } +} diff --git a/config/Migrations/20241114090547_CreateProducts.php b/config/Migrations/20241114090547_CreateProducts.php new file mode 100644 index 0000000..f7c6833 --- /dev/null +++ b/config/Migrations/20241114090547_CreateProducts.php @@ -0,0 +1,47 @@ +<?php +declare(strict_types=1); + +use Migrations\AbstractMigration; + +class CreateProducts extends AbstractMigration +{ + /** + * Change Method. + * + * More information on this method is available here: + * https://book.cakephp.org/phinx/0/en/migrations.html#the-change-method + * @return void + */ + public function change(): void + { + $table = $this->table('products', ['id' => false, 'primary_key' => ['id']]); + $table->addColumn('id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('name', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('product_category_id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('product_type_id', 'integer', [ + 'default' => null, + 'limit' => 11, + 'null' => false, + ]); + $table->addIndex('product_category_id'); + $table->addIndex('product_type_id'); + $table->addIndex([ + 'product_category_id', + 'name', + ], [ + 'name' => 'BY_NAME_AND_CATEGORY_ID', + 'unique' => true, + ]); + $table->create(); + } +} diff --git a/config/Migrations/20241115054627_CreateProductCategoryAttributes.php b/config/Migrations/20241115054627_CreateProductCategoryAttributes.php new file mode 100644 index 0000000..b229ec3 --- /dev/null +++ b/config/Migrations/20241115054627_CreateProductCategoryAttributes.php @@ -0,0 +1,55 @@ +<?php +declare(strict_types=1); + +use Migrations\AbstractMigration; + +class CreateProductCategoryAttributes extends AbstractMigration +{ + /** + * Change Method. + * + * More information on this method is available here: + * https://book.cakephp.org/phinx/0/en/migrations.html#the-change-method + * @return void + */ + public function change(): void + { + $table = $this->table('product_category_attributes', ['id' => false, 'primary_key' => ['id']]); + $table->addColumn('id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('name', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('product_category_id', 'uuid', [ + 'default' => null, + 'null' => true, + ]); + $table->addColumn('attribute_type_id', 'integer', [ + 'default' => null, + 'limit' => 11, + 'null' => false, + ]); + $table->addColumn('enabled', 'boolean', [ + 'default' => null, + 'null' => false, + ]); + $table->addIndex([ + 'product_category_id', + ], [ + 'name' => 'BY_PRODUCT_CATEGORY_ID', + 'unique' => false, + ]); + $table->addIndex([ + 'name', + 'product_category_id', + ], [ + 'name' => 'BY_NAME_AND_PRODUCT_CATEGORY_ID_UNIQUE', + 'unique' => true, + ]); + $table->create(); + } +} diff --git a/config/Migrations/20241115062613_CreateProductCategoryAttributeOptions.php b/config/Migrations/20241115062613_CreateProductCategoryAttributeOptions.php new file mode 100644 index 0000000..868f349 --- /dev/null +++ b/config/Migrations/20241115062613_CreateProductCategoryAttributeOptions.php @@ -0,0 +1,48 @@ +<?php +declare(strict_types=1); + +use Migrations\AbstractMigration; + +class CreateProductCategoryAttributeOptions extends AbstractMigration +{ + /** + * Change Method. + * + * More information on this method is available here: + * https://book.cakephp.org/phinx/0/en/migrations.html#the-change-method + * @return void + */ + public function change(): void + { + $table = $this->table('product_category_attribute_options', ['id' => false, 'primary_key' => ['id']]); + $table->addColumn('id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('product_category_attribute_id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('attribute_value', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('attribute_label', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('enabled', 'boolean', [ + 'default' => true, + 'null' => false, + ]); + $table->addIndex([ + 'product_category_attribute_id', + ], [ + 'name' => 'BY_PRODUCT_CATEGORY_ATTRIBUTE_ID', + 'unique' => false, + ]); + $table->create(); + } +} diff --git a/config/Migrations/20241122070040_CreateExternalProductCatalogs.php b/config/Migrations/20241122070040_CreateExternalProductCatalogs.php new file mode 100644 index 0000000..1382ee1 --- /dev/null +++ b/config/Migrations/20241122070040_CreateExternalProductCatalogs.php @@ -0,0 +1,52 @@ +<?php +declare(strict_types=1); + +use Migrations\AbstractMigration; + +class CreateExternalProductCatalogs extends AbstractMigration +{ + /** + * Change Method. + * + * More information on this method is available here: + * https://book.cakephp.org/phinx/0/en/migrations.html#the-change-method + * @return void + */ + public function change(): void + { + $table = $this->table('external_product_catalogs'); + $table->addColumn('product_catalog_id', 'uuid', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('base_url', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('api_url', 'string', [ + 'default' => null, + 'limit' => 255, + 'null' => false, + ]); + $table->addColumn('created', 'datetime', [ + 'default' => null, + 'null' => false, + ]); + $table->addColumn('deleted', 'datetime', [ + 'default' => null, + 'null' => true, + ]); + $table->addColumn('enabled', 'boolean', [ + 'default' => null, + 'null' => false, + ]); + $table->addIndex([ + 'product_catalog_id', + ], [ + 'name' => 'BY_PRODUCT_CATALOG_ID', + 'unique' => false, + ]); + $table->create(); + } +} diff --git a/config/app.example.php b/config/app.example.php new file mode 100644 index 0000000..488d47d --- /dev/null +++ b/config/app.example.php @@ -0,0 +1,28 @@ +<?php + +// The following configs can be globally configured, copy the array content over to your ROOT/config + +return [ + 'CakeProducts' => [ + /** + * internal CakeProducts settings - used in the source of truth/internal only system. + * Can optionally manage external catalogs + * + * - syncExternally - defaults to false - product catalogs can have 1 or more external catalogs linked to them + * which will receive changes to the catalogs and optionally allow for external API access. + * Will have no effect if true but no external catalogs have been added or none are enabled + */ + 'internal' => [ + 'enabled' => true, + /** + * syncExternally defaults to false - product catalogs can have 1 or more external catalogs linked to them + * which will receive changes to the catalogs and optionally allow for external API access. + * Will have no effect if true but no external catalogs have been added or none are enabled + */ + 'syncExternally' => false, + ], + 'external' => [ // product catalog settings for external use (as an API server to power an ecommerce site for example) + 'enabled' => false, + ], + ], +]; diff --git a/phpunit.xml.dist b/phpunit.xml.dist new file mode 100644 index 0000000..d9447ae --- /dev/null +++ b/phpunit.xml.dist @@ -0,0 +1,30 @@ +<?xml version="1.0" encoding="UTF-8"?> +<phpunit + colors="true" + processIsolation="false" + stopOnFailure="false" + bootstrap="tests/bootstrap.php" + > + <php> + <ini name="memory_limit" value="-1"/> + <ini name="apc.enable_cli" value="1"/> + </php> + + <!-- Add any additional test suites you want to run here --> + <testsuites> + <testsuite name="CakeProducts"> + <directory>tests/TestCase/</directory> + </testsuite> + </testsuites> + + <!-- Setup fixture extension --> + <extensions> + <bootstrap class="Cake\TestSuite\Fixture\Extension\PHPUnitExtension"/> + </extensions> + + <source> + <include> + <directory suffix=".php">src/</directory> + </include> + </source> +</phpunit> diff --git a/src/CakeProductsPlugin.php b/src/CakeProductsPlugin.php new file mode 100644 index 0000000..614d6fa --- /dev/null +++ b/src/CakeProductsPlugin.php @@ -0,0 +1,95 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts; + +use Cake\Console\CommandCollection; +use Cake\Core\BasePlugin; +use Cake\Core\ContainerInterface; +use Cake\Core\PluginApplicationInterface; +use Cake\Http\MiddlewareQueue; +use Cake\Routing\RouteBuilder; +use CakeProducts\Service\CatalogManagerServiceProvider; + +/** + * Plugin for CakeProducts + */ +class CakeProductsPlugin extends BasePlugin +{ + /** + * Load all the plugin configuration and bootstrap logic. + * + * The host application is provided as an argument. This allows you to load + * additional plugin dependencies, or attach events. + * + * @param \Cake\Core\PluginApplicationInterface $app The host application + * @return void + */ + public function bootstrap(PluginApplicationInterface $app): void + { + } + + /** + * Add routes for the plugin. + * + * If your plugin has many routes and you would like to isolate them into a separate file, + * you can create `$plugin/config/routes.php` and delete this method. + * + * @param \Cake\Routing\RouteBuilder $routes The route builder to update. + * @return void + */ + public function routes(RouteBuilder $routes): void + { + $routes->plugin( + 'CakeProducts', + ['path' => '/cake-products'], + function (RouteBuilder $builder) { + // Add custom routes here + + $builder->fallbacks(); + } + ); + parent::routes($routes); + } + + /** + * Add middleware for the plugin. + * + * @param \Cake\Http\MiddlewareQueue $middlewareQueue The middleware queue to update. + * @return \Cake\Http\MiddlewareQueue + */ + public function middleware(MiddlewareQueue $middlewareQueue): MiddlewareQueue + { + // Add your middlewares here + + return $middlewareQueue; + } + + /** + * Add commands for the plugin. + * + * @param \Cake\Console\CommandCollection $commands The command collection to update. + * @return \Cake\Console\CommandCollection + */ + public function console(CommandCollection $commands): CommandCollection + { + // Add your commands here + + $commands = parent::console($commands); + + return $commands; + } + + /** + * Register application container services. + * + * @param \Cake\Core\ContainerInterface $container The Container to update. + * @return void + * @link https://book.cakephp.org/4/en/development/dependency-injection.html#dependency-injection + */ + public function services(ContainerInterface $container): void + { + // Add your services here + $container->addServiceProvider(new CatalogManagerServiceProvider()); + } +} diff --git a/src/Controller/AppController.php b/src/Controller/AppController.php new file mode 100644 index 0000000..c63822f --- /dev/null +++ b/src/Controller/AppController.php @@ -0,0 +1,10 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Controller; + +use App\Controller\AppController as BaseController; + +class AppController extends BaseController +{ +} diff --git a/src/Controller/ExternalProductCatalogsController.php b/src/Controller/ExternalProductCatalogsController.php new file mode 100644 index 0000000..1683ce3 --- /dev/null +++ b/src/Controller/ExternalProductCatalogsController.php @@ -0,0 +1,110 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Controller; + +use Cake\Log\Log; +use CakeProducts\Controller\AppController; + +/** + * ExternalProductCatalogs Controller + * + * @property \CakeProducts\Model\Table\ExternalProductCatalogsTable $ExternalProductCatalogs + */ +class ExternalProductCatalogsController extends AppController +{ + /** + * Index method + * + * @return \Cake\Http\Response|null|void Renders view + */ + public function index() + { + $query = $this->ExternalProductCatalogs->find() + ->contain(['ProductCatalogs']); + $externalProductCatalogs = $this->paginate($query); + + $this->set(compact('externalProductCatalogs')); + } + + /** + * View method + * + * @param string|null $id External Product Catalog id. + * @return \Cake\Http\Response|null|void Renders view + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function view($id = null) + { + $externalProductCatalog = $this->ExternalProductCatalogs->get($id, contain: ['ProductCatalogs']); + $this->set(compact('externalProductCatalog')); + } + + /** + * Add method + * + * @return \Cake\Http\Response|null|void Redirects on successful add, renders view otherwise. + */ + public function add() + { + $externalProductCatalog = $this->ExternalProductCatalogs->newEmptyEntity(); + if ($this->request->is('post')) { + $externalProductCatalog = $this->ExternalProductCatalogs->patchEntity($externalProductCatalog, $this->request->getData()); + if ($this->ExternalProductCatalogs->save($externalProductCatalog)) { + $this->Flash->success(__('The external product catalog has been saved.')); + + return $this->redirect(['action' => 'index']); + } + Log::debug(print_r('$externalProductCatalog->getErrors() next - failed /add', true)); + Log::debug(print_r($externalProductCatalog->getErrors(), true)); + $this->Flash->error(__('The external product catalog could not be saved. Please, try again.')); + } + $productCatalogs = $this->ExternalProductCatalogs->ProductCatalogs->find('list', limit: 200)->all(); + $this->set(compact('externalProductCatalog', 'productCatalogs')); + } + + /** + * Edit method + * + * @param string|null $id External Product Catalog id. + * @return \Cake\Http\Response|null|void Redirects on successful edit, renders view otherwise. + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function edit($id = null) + { + $externalProductCatalog = $this->ExternalProductCatalogs->get($id, contain: []); + if ($this->request->is(['patch', 'post', 'put'])) { + $externalProductCatalog = $this->ExternalProductCatalogs->patchEntity($externalProductCatalog, $this->request->getData()); + if ($this->ExternalProductCatalogs->save($externalProductCatalog)) { + $this->Flash->success(__('The external product catalog has been saved.')); + + return $this->redirect(['action' => 'index']); + } + Log::debug(print_r('$externalProductCatalog->getErrors() next - failed /edit', true)); + Log::debug(print_r($externalProductCatalog->getErrors(), true)); + $this->Flash->error(__('The external product catalog could not be saved. Please, try again.')); + } + $productCatalogs = $this->ExternalProductCatalogs->ProductCatalogs->find('list', limit: 200)->all(); + $this->set(compact('externalProductCatalog', 'productCatalogs')); + } + + /** + * Delete method + * + * @param string|null $id External Product Catalog id. + * @return \Cake\Http\Response|null Redirects to index. + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function delete($id = null) + { + $this->request->allowMethod(['post', 'delete']); + $externalProductCatalog = $this->ExternalProductCatalogs->get($id); + if ($this->ExternalProductCatalogs->delete($externalProductCatalog)) { + $this->Flash->success(__('The external product catalog has been deleted.')); + } else { + $this->Flash->error(__('The external product catalog could not be deleted. Please, try again.')); + } + + return $this->redirect(['action' => 'index']); + } +} diff --git a/src/Controller/ProductCatalogsController.php b/src/Controller/ProductCatalogsController.php new file mode 100644 index 0000000..7026513 --- /dev/null +++ b/src/Controller/ProductCatalogsController.php @@ -0,0 +1,111 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Controller; + +use Cake\Datasource\Exception\RecordNotFoundException; +use Cake\Http\Response; +use Cake\Log\Log; +use CakeProducts\Controller\AppController; +use CakeProducts\Model\Table\ProductCatalogsTable; +use CakeProducts\Service\InternalCatalogManagerService; + +/** + * ProductCatalogs Controller + * + * @property ProductCatalogsTable $ProductCatalogs + */ +class ProductCatalogsController extends AppController +{ + /** + * Index method + * + * @return Response|null|void Renders view + */ + public function index() + { + $query = $this->ProductCatalogs->find(); + $productCatalogs = $this->paginate($query); + + $this->set(compact('productCatalogs')); + } + + /** + * View method + * + * @param string|null $id Product Catalog id. + * @return Response|null|void Renders view + * @throws RecordNotFoundException When record not found. + */ + public function view(InternalCatalogManagerService $catalogManagerService, $id = null) + { + $productCatalog = $catalogManagerService->getCatalog($id); + $this->set(compact('productCatalog')); + } + + /** + * Add method + * + * @return Response|null|void Redirects on successful add, renders view otherwise. + */ + public function add() + { + $productCatalog = $this->ProductCatalogs->newEmptyEntity(); + if ($this->request->is('post')) { + $productCatalog = $this->ProductCatalogs->patchEntity($productCatalog, $this->request->getData()); + if ($this->ProductCatalogs->save($productCatalog)) { + $this->Flash->success(__('The product catalog has been saved.')); + + return $this->redirect(['action' => 'index']); + } + Log::debug('failed to save new product catalog errors next'); + Log::debug(print_r('$productCatalog->getErrors()', true)); + Log::debug(print_r($productCatalog->getErrors(), true)); + + $this->Flash->error(__('The product catalog could not be saved. Please, try again.')); + } + $this->set(compact('productCatalog')); + } + + /** + * Edit method + * + * @param string|null $id Product Catalog id. + * @return Response|null|void Redirects on successful edit, renders view otherwise. + * @throws RecordNotFoundException When record not found. + */ + public function edit($id = null) + { + $productCatalog = $this->ProductCatalogs->get($id, contain: []); + if ($this->request->is(['patch', 'post', 'put'])) { + $productCatalog = $this->ProductCatalogs->patchEntity($productCatalog, $this->request->getData()); + if ($this->ProductCatalogs->save($productCatalog)) { + $this->Flash->success(__('The product catalog has been saved.')); + + return $this->redirect(['action' => 'index']); + } + $this->Flash->error(__('The product catalog could not be saved. Please, try again.')); + } + $this->set(compact('productCatalog')); + } + + /** + * Delete method + * + * @param string|null $id Product Catalog id. + * @return Response|null Redirects to index. + * @throws RecordNotFoundException When record not found. + */ + public function delete($id = null) + { + $this->request->allowMethod(['post', 'delete']); + $productCatalog = $this->ProductCatalogs->get($id); + if ($this->ProductCatalogs->delete($productCatalog)) { + $this->Flash->success(__('The product catalog has been deleted.')); + } else { + $this->Flash->error(__('The product catalog could not be deleted. Please, try again.')); + } + + return $this->redirect(['action' => 'index']); + } +} diff --git a/src/Controller/ProductCategoriesController.php b/src/Controller/ProductCategoriesController.php new file mode 100644 index 0000000..22ca85c --- /dev/null +++ b/src/Controller/ProductCategoriesController.php @@ -0,0 +1,115 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Controller; + +use Cake\Log\Log; +use CakeProducts\Controller\AppController; +use CakeProducts\Service\InternalCatalogManagerService; + +/** + * ProductCategories Controller + * + * @property \CakeProducts\Model\Table\ProductCategoriesTable $ProductCategories + */ +class ProductCategoriesController extends AppController +{ + /** + * Index method + * + * @return \Cake\Http\Response|null|void Renders view + */ + public function index() + { + $query = $this->ProductCategories->find() + ->contain(['ProductCatalogs', 'ParentProductCategories']); + $productCategories = $this->paginate($query); + + $this->set(compact('productCategories')); + } + + /** + * View method + * + * @param string|null $id Product Category id. + * @return \Cake\Http\Response|null|void Renders view + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function view(InternalCatalogManagerService $catalogManagerService, $id = null) + { + $productCategory = $catalogManagerService->getCategory($id); + $this->set(compact('productCategory')); + } + + /** + * Add method + * + * @return \Cake\Http\Response|null|void Redirects on successful add, renders view otherwise. + */ + public function add(InternalCatalogManagerService $catalogManagerService) + { + $productCategory = $this->ProductCategories->newEmptyEntity(); + if ($this->request->is('post')) { + $postData = $this->request->getData(); + if ($this->request->getSession()->read('Auth.User.id')) { + $postData['created_by'] = $this->request->getSession()->read('Auth.User.id'); + } + $result = $catalogManagerService->createNewCategory($productCategory, $postData); + Log::debug(print_r('$result from createNewCategory', true)); + Log::debug(print_r($result, true)); + if ($result['result']) { + $this->Flash->success(__('The product category has been saved.')); + + return $this->redirect(['action' => 'index']); + } + $this->Flash->error(__('The product category could not be saved. Please, try again.')); + } + $productCatalogs = $this->ProductCategories->ProductCatalogs->find('list', limit: 200)->all(); + $parentProductCategories = $this->ProductCategories->ParentProductCategories->find('list', limit: 200)->all(); + $this->set(compact('productCategory', 'productCatalogs', 'parentProductCategories')); + } + + /** + * Edit method + * + * @param string|null $id Product Category id. + * @return \Cake\Http\Response|null|void Redirects on successful edit, renders view otherwise. + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function edit(InternalCatalogManagerService $catalogManagerService, $id = null) + { + $productCategory = $this->ProductCategories->get($id, contain: []); + if ($this->request->is(['patch', 'post', 'put'])) { + $productCategory = $this->ProductCategories->patchEntity($productCategory, $this->request->getData()); + if ($this->ProductCategories->save($productCategory)) { + $this->Flash->success(__('The product category has been saved.')); + + return $this->redirect(['action' => 'index']); + } + $this->Flash->error(__('The product category could not be saved. Please, try again.')); + } + $productCatalogs = $this->ProductCategories->ProductCatalogs->find('list', limit: 200)->all(); + $parentProductCategories = $this->ProductCategories->ParentProductCategories->find('list', limit: 200)->all(); + $this->set(compact('productCategory', 'productCatalogs', 'parentProductCategories')); + } + + /** + * Delete method + * + * @param string|null $id Product Category id. + * @return \Cake\Http\Response|null Redirects to index. + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function delete($id = null) + { + $this->request->allowMethod(['post', 'delete']); + $productCategory = $this->ProductCategories->get($id); + if ($this->ProductCategories->delete($productCategory)) { + $this->Flash->success(__('The product category has been deleted.')); + } else { + $this->Flash->error(__('The product category could not be deleted. Please, try again.')); + } + + return $this->redirect(['action' => 'index']); + } +} diff --git a/src/Controller/ProductCategoryAttributeOptionsController.php b/src/Controller/ProductCategoryAttributeOptionsController.php new file mode 100644 index 0000000..b166725 --- /dev/null +++ b/src/Controller/ProductCategoryAttributeOptionsController.php @@ -0,0 +1,48 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Controller; + +use Cake\Log\Log; +use CakeProducts\Controller\AppController; + +/** + * ProductCategoryAttributeOptions Controller + * + * @property \CakeProducts\Model\Table\ProductCategoryAttributeOptionsTable $ProductCategoryAttributeOptions + */ +class ProductCategoryAttributeOptionsController extends AppController +{ + /** + * Add method + * + * @return \Cake\Http\Response|null|void Redirects on successful add, renders view otherwise. + */ + public function add() + { + Log::debug('inside product category attribute options controller add'); + + $productCategoryAttributeOption = $this->ProductCategoryAttributeOptions->newEmptyEntity(); + $this->set(compact('productCategoryAttributeOption')); + } + + /** + * Delete method + * + * @param string|null $id Product Category Attribute Option id. + * @return \Cake\Http\Response|null Redirects to index. + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function delete($id = null) + { + $this->request->allowMethod(['post', 'delete']); + $productCategoryAttributeOption = $this->ProductCategoryAttributeOptions->get($id); + if ($this->ProductCategoryAttributeOptions->delete($productCategoryAttributeOption)) { + $this->Flash->success(__('The product category attribute option has been deleted.')); + } else { + $this->Flash->error(__('The product category attribute option could not be deleted. Please, try again.')); + } + + return $this->redirect(['controller' => 'ProductCategoryAttributes', 'action' => 'view', $productCategoryAttributeOption->product_category_attribute_id]); + } +} diff --git a/src/Controller/ProductCategoryAttributesController.php b/src/Controller/ProductCategoryAttributesController.php new file mode 100644 index 0000000..4e83d89 --- /dev/null +++ b/src/Controller/ProductCategoryAttributesController.php @@ -0,0 +1,144 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Controller; + +use Cake\Datasource\Exception\RecordNotFoundException; +use Cake\Http\Response; +use Cake\Log\Log; +use CakeProducts\Controller\AppController; +use CakeProducts\Model\Enum\ProductCategoryAttributeTypeId; +use CakeProducts\Model\Table\ProductCategoryAttributesTable; + +/** + * ProductCategoryAttributes Controller + * + * @property ProductCategoryAttributesTable $ProductCategoryAttributes + */ +class ProductCategoryAttributesController extends AppController +{ + /** + * Index method + * + * @return Response|null|void Renders view + */ + public function index() + { + $query = $this->ProductCategoryAttributes->find() + ->contain(['ProductCategories']); + $productCategoryAttributes = $this->paginate($query); + + $this->set(compact('productCategoryAttributes')); + } + + /** + * View method + * + * @param string|null $id Product Category Attribute id. + * @return Response|null|void Renders view + * @throws RecordNotFoundException When record not found. + */ + public function view($id = null) + { + $productCategoryAttribute = $this->ProductCategoryAttributes->get($id, contain: [ + 'ProductCategories', + 'ProductCategoryAttributeOptions', + ]); + $this->set(compact('productCategoryAttribute')); + } + + /** + * Add method + * + * @return Response|null|void Redirects on successful add, renders view otherwise. + */ + public function add() + { + $productCategoryAttribute = $this->ProductCategoryAttributes->newEmptyEntity(); + if ($this->request->is('post')) { + $postData = $this->request->getData(); + $saveOptions = [ + 'associated' => ['ProductCategoryAttributeOptions'], + ]; + Log::debug(print_r('$postData', true)); + Log::debug(print_r($postData, true)); +// if ($this->request->getData('attribute_type_id') != ProductCategoryAttributeTypeId::Constrained) { +// $saveOptions['associated'] = []; +// $postData['product_category_attribute_options'] = []; +// } + Log::debug(print_r('$postData', true)); + Log::debug(print_r($postData, true)); + $productCategoryAttribute = $this->ProductCategoryAttributes->patchEntity($productCategoryAttribute, $postData, $saveOptions); + if ($this->ProductCategoryAttributes->save($productCategoryAttribute, $saveOptions)) { + $this->Flash->success(__('The product category attribute has been saved.')); + + return $this->redirect(['action' => 'index']); + } + Log::debug('failed to save new product category attribute errors next'); + Log::debug(print_r('$productCategoryAttribute->getErrors()', true)); + Log::debug(print_r($productCategoryAttribute->getErrors(), true)); + $this->Flash->error(__('The product category attribute could not be saved. Please, try again.')); + } + $productCategories = $this->ProductCategoryAttributes->ProductCategories->find('list', limit: 200)->all(); + $this->set(compact('productCategoryAttribute', 'productCategories')); + } + + /** + * Edit method + * + * @param string|null $id Product Category Attribute id. + * @return Response|null|void Redirects on successful edit, renders view otherwise. + * @throws RecordNotFoundException When record not found. + */ + public function edit($id = null) + { + $productCategoryAttribute = $this->ProductCategoryAttributes->get($id, contain: ['ProductCategoryAttributeOptions']); + if ($this->request->is(['patch', 'post', 'put'])) { + $postData = $this->request->getData(); + $saveOptions = [ + 'associated' => ['ProductCategoryAttributeOptions'], + ]; + Log::debug(print_r('$postData', true)); + Log::debug(print_r($postData, true)); +// if ($this->request->getData('attribute_type_id') != ProductCategoryAttributeTypeId::Constrained) { +// $saveOptions['associated'] = []; +// $postData['product_category_attribute_options'] = []; +// } + Log::debug(print_r('$postData', true)); + Log::debug(print_r($postData, true)); + $productCategoryAttribute = $this->ProductCategoryAttributes->patchEntity($productCategoryAttribute, $postData, $saveOptions); + + if ($this->ProductCategoryAttributes->save($productCategoryAttribute, $saveOptions)) { + $this->Flash->success(__('The product category attribute has been saved.')); + + return $this->redirect(['action' => 'index']); + } + Log::debug('failed to save product category attribute on edit errors next'); + Log::debug(print_r('$productCategoryAttribute->getErrors()', true)); + Log::debug(print_r($productCategoryAttribute->getErrors(), true)); + $this->Flash->error(__('The product category attribute could not be saved. Please, try again.')); + } + $productCategories = $this->ProductCategoryAttributes->ProductCategories->find('list', limit: 200)->all(); + $this->set(compact('productCategoryAttribute', 'productCategories')); + } + + /** + * Delete method + * + * @param string|null $id Product Category Attribute id. + * @return Response|null Redirects to index. + * @throws RecordNotFoundException When record not found. + */ + public function delete($id = null) + { + $this->request->allowMethod(['post', 'delete']); + $productCategoryAttribute = $this->ProductCategoryAttributes->get($id); + if ($this->ProductCategoryAttributes->delete($productCategoryAttribute)) { + $this->Flash->success(__('The product category attribute has been deleted.')); + } else { + $this->Flash->error(__('The product category attribute could not be deleted. Please, try again.')); + } + + return $this->redirect(['action' => 'index']); + } +} diff --git a/src/Controller/ProductsController.php b/src/Controller/ProductsController.php new file mode 100644 index 0000000..d3648a2 --- /dev/null +++ b/src/Controller/ProductsController.php @@ -0,0 +1,110 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Controller; + +use Cake\Log\Log; +use CakeProducts\Controller\AppController; + +/** + * Products Controller + * + * @property \CakeProducts\Model\Table\ProductsTable $Products + */ +class ProductsController extends AppController +{ + /** + * Index method + * + * @return \Cake\Http\Response|null|void Renders view + */ + public function index() + { + $query = $this->Products->find() + ->contain(['ProductCategories']); + $products = $this->paginate($query); + + $this->set(compact('products')); + } + + /** + * View method + * + * @param string|null $id Product id. + * @return \Cake\Http\Response|null|void Renders view + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function view($id = null) + { + $product = $this->Products->get($id, contain: ['ProductCategories']); + $this->set(compact('product')); + } + + /** + * Add method + * + * @return \Cake\Http\Response|null|void Redirects on successful add, renders view otherwise. + */ + public function add() + { + $product = $this->Products->newEmptyEntity(); + if ($this->request->is('post')) { + $product = $this->Products->patchEntity($product, $this->request->getData()); + if ($this->Products->save($product)) { + $this->Flash->success(__('The product has been saved.')); + + return $this->redirect(['action' => 'index']); + } + Log::debug(print_r('$product->getErrors() next - failed in products/add', true)); + Log::debug(print_r($product->getErrors(), true)); + $this->Flash->error(__('The product could not be saved. Please, try again.')); + } + $productCategories = $this->Products->ProductCategories->find('list', limit: 200)->all(); + $this->set(compact('product', 'productCategories')); + } + + /** + * Edit method + * + * @param string|null $id Product id. + * @return \Cake\Http\Response|null|void Redirects on successful edit, renders view otherwise. + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function edit($id = null) + { + $product = $this->Products->get($id, contain: []); + if ($this->request->is(['patch', 'post', 'put'])) { + $product = $this->Products->patchEntity($product, $this->request->getData()); + if ($this->Products->save($product)) { + $this->Flash->success(__('The product has been saved.')); + + return $this->redirect(['action' => 'index']); + } + Log::debug(print_r('$product->getErrors() next - failed in products/edit', true)); + Log::debug(print_r($product->getErrors(), true)); + $this->Flash->error(__('The product could not be saved. Please, try again.')); + } + $productCategories = $this->Products->ProductCategories->find('list', limit: 200)->all(); + $this->set(compact('product', 'productCategories')); + } + + /** + * Delete method + * + * @param string|null $id Product id. + * @return \Cake\Http\Response|null Redirects to index. + * @throws \Cake\Datasource\Exception\RecordNotFoundException When record not found. + */ + public function delete($id = null) + { + $this->request->allowMethod(['post', 'delete']); + $product = $this->Products->get($id); + if ($this->Products->delete($product)) { + $this->Flash->success(__('The product has been deleted.')); + } else { + $this->Flash->error(__('The product could not be deleted. Please, try again.')); + } + + return $this->redirect(['action' => 'index']); + } +} diff --git a/src/Model/Entity/ExternalProductCatalog.php b/src/Model/Entity/ExternalProductCatalog.php new file mode 100644 index 0000000..9b03a7c --- /dev/null +++ b/src/Model/Entity/ExternalProductCatalog.php @@ -0,0 +1,41 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Entity; + +use Cake\ORM\Entity; + +/** + * ExternalProductCatalog Entity + * + * @property int $id + * @property string $product_catalog_id + * @property string $base_url + * @property string $api_url + * @property \Cake\I18n\DateTime $created + * @property \Cake\I18n\DateTime|null $deleted + * @property bool $enabled + * + * @property \CakeProducts\Model\Entity\ProductCatalog $product_catalog + */ +class ExternalProductCatalog extends Entity +{ + /** + * Fields that can be mass assigned using newEntity() or patchEntity(). + * + * Note that when '*' is set to true, this allows all unspecified fields to + * be mass assigned. For security purposes, it is advised to set '*' to false + * (or remove it), and explicitly make individual fields accessible as needed. + * + * @var array<string, bool> + */ + protected array $_accessible = [ + 'product_catalog_id' => true, + 'base_url' => true, + 'api_url' => true, + 'created' => true, + 'deleted' => true, + 'enabled' => true, + 'product_catalog' => true, + ]; +} diff --git a/src/Model/Entity/Product.php b/src/Model/Entity/Product.php new file mode 100644 index 0000000..6cac6aa --- /dev/null +++ b/src/Model/Entity/Product.php @@ -0,0 +1,35 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Entity; + +use Cake\ORM\Entity; + +/** + * Product Entity + * + * @property string $id + * @property string $name + * @property string $product_category_id + * @property \CakeProducts\Model\Enum\ProductProductTypeId $product_type_id + * + * @property \CakeProducts\Model\Entity\ProductCategory $product_category + */ +class Product extends Entity +{ + /** + * Fields that can be mass assigned using newEntity() or patchEntity(). + * + * Note that when '*' is set to true, this allows all unspecified fields to + * be mass assigned. For security purposes, it is advised to set '*' to false + * (or remove it), and explicitly make individual fields accessible as needed. + * + * @var array<string, bool> + */ + protected array $_accessible = [ + 'name' => true, + 'product_category_id' => true, + 'product_type_id' => true, + 'product_category' => true, + ]; +} diff --git a/src/Model/Entity/ProductCatalog.php b/src/Model/Entity/ProductCatalog.php new file mode 100644 index 0000000..f037960 --- /dev/null +++ b/src/Model/Entity/ProductCatalog.php @@ -0,0 +1,37 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Entity; + +use Cake\ORM\Entity; + +/** + * ProductCatalog Entity + * + * @property string $id + * @property string $name + * @property string|null $catalog_description + * @property bool $enabled + * + * @property \CakeProducts\Model\Entity\ProductCategory[] $product_categories + * @property \CakeProducts\Model\Entity\ExternalProductCatalog[] $external_product_catalogs + */ +class ProductCatalog extends Entity +{ + /** + * Fields that can be mass assigned using newEntity() or patchEntity(). + * + * Note that when '*' is set to true, this allows all unspecified fields to + * be mass assigned. For security purposes, it is advised to set '*' to false + * (or remove it), and explicitly make individual fields accessible as needed. + * + * @var array<string, bool> + */ + protected array $_accessible = [ + 'name' => true, + 'catalog_description' => true, + 'enabled' => true, + 'product_categories' => true, + 'external_product_catalogs' => true, + ]; +} diff --git a/src/Model/Entity/ProductCategory.php b/src/Model/Entity/ProductCategory.php new file mode 100644 index 0000000..c7e89ee --- /dev/null +++ b/src/Model/Entity/ProductCategory.php @@ -0,0 +1,49 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Entity; + +use Cake\ORM\Entity; + +/** + * ProductCategory Entity + * + * @property int $id + * @property string $internal_id + * @property string $product_catalog_id + * @property string $name + * @property string|null $category_description + * @property int|null $parent_id + * @property int $lft + * @property int $rght + * @property bool $enabled + * + * @property \CakeProducts\Model\Entity\ProductCatalog $product_catalog + * @property \CakeProducts\Model\Entity\ParentProductCategory $parent_product_category + * @property \CakeProducts\Model\Entity\ChildProductCategory[] $child_product_categories + */ +class ProductCategory extends Entity +{ + /** + * Fields that can be mass assigned using newEntity() or patchEntity(). + * + * Note that when '*' is set to true, this allows all unspecified fields to + * be mass assigned. For security purposes, it is advised to set '*' to false + * (or remove it), and explicitly make individual fields accessible as needed. + * + * @var array<string, bool> + */ + protected array $_accessible = [ + 'product_catalog_id' => true, + 'internal_id' => true, + 'name' => true, + 'category_description' => true, + 'parent_id' => true, + 'lft' => true, + 'rght' => true, + 'enabled' => true, + 'product_catalog' => true, + 'parent_product_category' => true, + 'child_product_categories' => true, + ]; +} diff --git a/src/Model/Entity/ProductCategoryAttribute.php b/src/Model/Entity/ProductCategoryAttribute.php new file mode 100644 index 0000000..e056a73 --- /dev/null +++ b/src/Model/Entity/ProductCategoryAttribute.php @@ -0,0 +1,39 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Entity; + +use Cake\ORM\Entity; + +/** + * ProductCategoryAttribute Entity + * + * @property string $id + * @property string $name + * @property string|null $product_category_id + * @property int $attribute_type_id + * @property bool $enabled + * + * @property ProductCategory $product_category + * @property ProductCategoryAttributeOption[] $product_category_attribute_options + */ +class ProductCategoryAttribute extends Entity +{ + /** + * Fields that can be mass assigned using newEntity() or patchEntity(). + * + * Note that when '*' is set to true, this allows all unspecified fields to + * be mass assigned. For security purposes, it is advised to set '*' to false + * (or remove it), and explicitly make individual fields accessible as needed. + * + * @var array<string, bool> + */ + protected array $_accessible = [ + 'name' => true, + 'product_category_id' => true, + 'attribute_type_id' => true, + 'enabled' => true, + 'product_category' => true, + 'product_category_attribute_options' => true, + ]; +} diff --git a/src/Model/Entity/ProductCategoryAttributeOption.php b/src/Model/Entity/ProductCategoryAttributeOption.php new file mode 100644 index 0000000..1e23953 --- /dev/null +++ b/src/Model/Entity/ProductCategoryAttributeOption.php @@ -0,0 +1,37 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Entity; + +use Cake\ORM\Entity; + +/** + * ProductCategoryAttributeOption Entity + * + * @property string $id + * @property string $product_category_attribute_id + * @property string $attribute_value + * @property string $attribute_label + * @property bool $enabled + * + * @property ProductCategoryAttribute $product_category_attribute + */ +class ProductCategoryAttributeOption extends Entity +{ + /** + * Fields that can be mass assigned using newEntity() or patchEntity(). + * + * Note that when '*' is set to true, this allows all unspecified fields to + * be mass assigned. For security purposes, it is advised to set '*' to false + * (or remove it), and explicitly make individual fields accessible as needed. + * + * @var array<string, bool> + */ + protected array $_accessible = [ + 'product_category_attribute_id' => true, + 'attribute_value' => true, + 'attribute_label' => true, + 'enabled' => true, + 'product_category_attribute' => true, + ]; +} diff --git a/src/Model/Enum/ProductCategoryAttributeTypeId.php b/src/Model/Enum/ProductCategoryAttributeTypeId.php new file mode 100644 index 0000000..2b024c9 --- /dev/null +++ b/src/Model/Enum/ProductCategoryAttributeTypeId.php @@ -0,0 +1,23 @@ +<?php +namespace CakeProducts\Model\Enum; + +use Cake\Database\Type\EnumLabelInterface; +use Tools\Model\Enum\EnumOptionsTrait; + +enum ProductCategoryAttributeTypeId: int implements EnumLabelInterface +{ + use EnumOptionsTrait; + + case Constrained = 1; + case Text = 2; + case Integer = 3; + + public function label(): string + { + return match($this) { + self::Constrained => 'Constrained', + self::Text => 'Text', + self::Integer => 'Integer' + }; + } +} diff --git a/src/Model/Enum/ProductProductTypeId.php b/src/Model/Enum/ProductProductTypeId.php new file mode 100644 index 0000000..4e220fb --- /dev/null +++ b/src/Model/Enum/ProductProductTypeId.php @@ -0,0 +1,23 @@ +<?php +namespace CakeProducts\Model\Enum; + +use Cake\Database\Type\EnumLabelInterface; +use Tools\Model\Enum\EnumOptionsTrait; + +enum ProductProductTypeId: int implements EnumLabelInterface +{ + use EnumOptionsTrait; + + case Service = 1; + case Product = 2; + case Consumable = 3; + + public function label(): string + { + return match($this) { + self::Service => 'Service', + self::Product => 'Product', + self::Consumable => 'Consumable' + }; + } +} diff --git a/src/Model/Table/ExternalProductCatalogsTable.php b/src/Model/Table/ExternalProductCatalogsTable.php new file mode 100644 index 0000000..503d351 --- /dev/null +++ b/src/Model/Table/ExternalProductCatalogsTable.php @@ -0,0 +1,115 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Table; + +use Cake\Datasource\EntityInterface; +use Cake\Datasource\ResultSetInterface; +use Cake\ORM\Association\BelongsTo; +use Cake\ORM\Behavior\TimestampBehavior; +use Cake\ORM\Query\SelectQuery; +use Cake\ORM\RulesChecker; +use Cake\ORM\Table; +use Cake\Validation\Validator; +use CakeProducts\Model\Entity\ExternalProductCatalog; +use Closure; +use Psr\SimpleCache\CacheInterface; + +/** + * ExternalProductCatalogs Model + * + * @property ProductCatalogsTable&BelongsTo $ProductCatalogs + * + * @method ExternalProductCatalog newEmptyEntity() + * @method ExternalProductCatalog newEntity(array $data, array $options = []) + * @method array<ExternalProductCatalog> newEntities(array $data, array $options = []) + * @method ExternalProductCatalog get(mixed $primaryKey, array|string $finder = 'all', CacheInterface|string|null $cache = null, Closure|string|null $cacheKey = null, mixed ...$args) + * @method ExternalProductCatalog findOrCreate($search, ?callable $callback = null, array $options = []) + * @method ExternalProductCatalog patchEntity(EntityInterface $entity, array $data, array $options = []) + * @method array<ExternalProductCatalog> patchEntities(iterable $entities, array $data, array $options = []) + * @method ExternalProductCatalog|false save(EntityInterface $entity, array $options = []) + * @method ExternalProductCatalog saveOrFail(EntityInterface $entity, array $options = []) + * @method iterable<ExternalProductCatalog>|ResultSetInterface<ExternalProductCatalog>|false saveMany(iterable $entities, array $options = []) + * @method iterable<ExternalProductCatalog>|ResultSetInterface<ExternalProductCatalog> saveManyOrFail(iterable $entities, array $options = []) + * @method iterable<ExternalProductCatalog>|ResultSetInterface<ExternalProductCatalog>|false deleteMany(iterable $entities, array $options = []) + * @method iterable<ExternalProductCatalog>|ResultSetInterface<ExternalProductCatalog> deleteManyOrFail(iterable $entities, array $options = []) + * + * @mixin TimestampBehavior + */ +class ExternalProductCatalogsTable extends Table +{ + /** + * Initialize method + * + * @param array<string, mixed> $config The configuration for the Table. + * @return void + */ + public function initialize(array $config): void + { + parent::initialize($config); + + $this->setTable('external_product_catalogs'); + $this->setDisplayField('base_url'); + $this->setPrimaryKey('id'); + + $this->addBehavior('Timestamp'); + + $this->belongsTo('ProductCatalogs', [ + 'foreignKey' => 'product_catalog_id', + 'joinType' => 'INNER', + 'className' => 'CakeProducts.ProductCatalogs', + ]); + } + + /** + * Default validation rules. + * + * @param Validator $validator Validator instance. + * @return Validator + */ + public function validationDefault(Validator $validator): Validator + { + $validator + ->uuid('product_catalog_id') + ->notEmptyString('product_catalog_id'); + + $validator + ->scalar('base_url') + ->maxLength('base_url', 255) + ->requirePresence('base_url', 'create') + ->notEmptyString('base_url'); +// ->url('base_url'); + + $validator + ->scalar('api_url') + ->maxLength('api_url', 255) + ->requirePresence('api_url', 'create') + ->notEmptyString('api_url'); +// ->url('api_url'); + + $validator + ->dateTime('deleted') + ->allowEmptyDateTime('deleted'); + + $validator + ->boolean('enabled') + ->requirePresence('enabled', 'create') + ->notEmptyString('enabled'); + + return $validator; + } + + /** + * Returns a rules checker object that will be used for validating + * application integrity. + * + * @param RulesChecker $rules The rules object to be modified. + * @return RulesChecker + */ + public function buildRules(RulesChecker $rules): RulesChecker + { + $rules->add($rules->existsIn(['product_catalog_id'], 'ProductCatalogs'), ['errorField' => 'product_catalog_id']); + + return $rules; + } +} diff --git a/src/Model/Table/ProductCatalogsTable.php b/src/Model/Table/ProductCatalogsTable.php new file mode 100644 index 0000000..f7c4919 --- /dev/null +++ b/src/Model/Table/ProductCatalogsTable.php @@ -0,0 +1,99 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Table; + +use Cake\Core\Configure; +use Cake\Datasource\EntityInterface; +use Cake\Datasource\ResultSetInterface; +use Cake\ORM\Query\SelectQuery; +use Cake\ORM\RulesChecker; +use Cake\ORM\Table; +use Cake\Validation\Validator; +use CakeProducts\Model\Entity\ProductCatalog; +use Closure; +use Psr\SimpleCache\CacheInterface; + +/** + * ProductCatalogs Model + * + * @method ProductCatalog newEmptyEntity() + * @method ProductCatalog newEntity(array $data, array $options = []) + * @method array<ProductCatalog> newEntities(array $data, array $options = []) + * @method ProductCatalog get(mixed $primaryKey, array|string $finder = 'all', CacheInterface|string|null $cache = null, Closure|string|null $cacheKey = null, mixed ...$args) + * @method ProductCatalog findOrCreate($search, ?callable $callback = null, array $options = []) + * @method ProductCatalog patchEntity(EntityInterface $entity, array $data, array $options = []) + * @method array<ProductCatalog> patchEntities(iterable $entities, array $data, array $options = []) + * @method ProductCatalog|false save(EntityInterface $entity, array $options = []) + * @method ProductCatalog saveOrFail(EntityInterface $entity, array $options = []) + * @method iterable<ProductCatalog>|ResultSetInterface<ProductCatalog>|false saveMany(iterable $entities, array $options = []) + * @method iterable<ProductCatalog>|ResultSetInterface<ProductCatalog> saveManyOrFail(iterable $entities, array $options = []) + * @method iterable<ProductCatalog>|ResultSetInterface<ProductCatalog>|false deleteMany(iterable $entities, array $options = []) + * @method iterable<ProductCatalog>|ResultSetInterface<ProductCatalog> deleteManyOrFail(iterable $entities, array $options = []) + */ +class ProductCatalogsTable extends Table +{ + /** + * Initialize method + * + * @param array<string, mixed> $config The configuration for the Table. + * @return void + */ + public function initialize(array $config): void + { + parent::initialize($config); + + $this->setTable('product_catalogs'); + $this->setDisplayField('name'); + $this->setPrimaryKey('id'); + + $this->hasMany('ProductCategories', [ + 'className' => 'CakeProducts.ProductCategories', + ]); + $this->hasMany('ExternalProductCatalogs', [ + 'className' => 'CakeProducts.ExternalProductCatalogs', + ]); + } + + /** + * Default validation rules. + * + * @param Validator $validator Validator instance. + * @return Validator + */ + public function validationDefault(Validator $validator): Validator + { + $validator + ->scalar('name') + ->maxLength('name', 255) + ->requirePresence('name', 'create') + ->notEmptyString('name') + ->add('name', 'unique', ['rule' => 'validateUnique', 'provider' => 'table']); + + $validator + ->scalar('catalog_description') + ->maxLength('catalog_description', 255) + ->allowEmptyString('catalog_description'); + + $validator + ->boolean('enabled') + ->requirePresence('enabled', 'create') + ->notEmptyString('enabled'); + + return $validator; + } + + /** + * Returns a rules checker object that will be used for validating + * application integrity. + * + * @param RulesChecker $rules The rules object to be modified. + * @return RulesChecker + */ + public function buildRules(RulesChecker $rules): RulesChecker + { + $rules->add($rules->isUnique(['name']), ['errorField' => 'name']); + + return $rules; + } +} diff --git a/src/Model/Table/ProductCategoriesTable.php b/src/Model/Table/ProductCategoriesTable.php new file mode 100644 index 0000000..f2af56f --- /dev/null +++ b/src/Model/Table/ProductCategoriesTable.php @@ -0,0 +1,164 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Table; + +use Cake\Datasource\EntityInterface; +use Cake\Datasource\ResultSetInterface; +use Cake\ORM\Association\BelongsTo; +use Cake\ORM\Association\HasMany; +use Cake\ORM\Behavior\TreeBehavior; +use Cake\ORM\Query\SelectQuery; +use Cake\ORM\RulesChecker; +use Cake\ORM\Table; +use Cake\Validation\Validator; +use CakeProducts\Model\Entity\ProductCategory; +use Closure; +use Psr\SimpleCache\CacheInterface; + +/** + * ProductCategories Model + * + * @property ProductCatalogsTable&BelongsTo $ProductCatalogs + * @property ProductCategoriesTable&BelongsTo $ParentProductCategories + * @property ProductCategoriesTable&HasMany $ChildProductCategories + * + * @method ProductCategory newEmptyEntity() + * @method ProductCategory newEntity(array $data, array $options = []) + * @method array<ProductCategory> newEntities(array $data, array $options = []) + * @method ProductCategory get(mixed $primaryKey, array|string $finder = 'all', CacheInterface|string|null $cache = null, Closure|string|null $cacheKey = null, mixed ...$args) + * @method ProductCategory findOrCreate($search, ?callable $callback = null, array $options = []) + * @method ProductCategory patchEntity(EntityInterface $entity, array $data, array $options = []) + * @method array<ProductCategory> patchEntities(iterable $entities, array $data, array $options = []) + * @method ProductCategory saveOrFail(EntityInterface $entity, array $options = []) + * @method iterable<ProductCategory>|ResultSetInterface<ProductCategory>|false saveMany(iterable $entities, array $options = []) + * @method iterable<ProductCategory>|ResultSetInterface<ProductCategory> saveManyOrFail(iterable $entities, array $options = []) + * @method iterable<ProductCategory>|ResultSetInterface<ProductCategory>|false deleteMany(iterable $entities, array $options = []) + * @method iterable<ProductCategory>|ResultSetInterface<ProductCategory> deleteManyOrFail(iterable $entities, array $options = []) + * + * @mixin TreeBehavior + */ +class ProductCategoriesTable extends Table +{ + /** + * Current scope for Tree behavior - per catalog + * + * @var string + */ + protected $treeCatalogId; + + /** + * Initialize method + * + * @param array<string, mixed> $config The configuration for the Table. + * @return void + */ + public function initialize(array $config): void + { + parent::initialize($config); + $this->treeCatalogId = 1; + + $this->setTable('product_categories'); + $this->setDisplayField('name'); + $this->setPrimaryKey('id'); + + $this->addBehavior('Tree'); + + $this->belongsTo('ProductCatalogs', [ + 'foreignKey' => 'product_catalog_id', + 'joinType' => 'INNER', + 'className' => 'CakeProducts.ProductCatalogs', + ]); + $this->belongsTo('ParentProductCategories', [ + 'className' => 'CakeProducts.ProductCategories', + 'foreignKey' => 'parent_id', + ]); + $this->hasMany('ChildProductCategories', [ + 'className' => 'CakeProducts.ProductCategories', + 'foreignKey' => 'parent_id', + ]); + $this->hasMany('ProductCategoryAttributes', [ + 'foreignKey' => 'product_category_id', + 'bindingKey' => 'internal_id', + 'className' => 'CakeProducts.ProductCategoryAttributes', + ]); + $this->behaviors()->Tree->setConfig('scope', ['product_catalog_id' => $this->treeCatalogId]); + } + + /** + * Default validation rules. + * + * @param Validator $validator Validator instance. + * @return Validator + */ + public function validationDefault(Validator $validator): Validator + { + $validator + ->uuid('product_catalog_id') + ->notEmptyString('product_catalog_id'); + + $validator + ->scalar('name') + ->maxLength('name', 255) + ->requirePresence('name', 'create') + ->notEmptyString('name'); + + $validator + ->scalar('category_description') + ->allowEmptyString('category_description'); + + $validator + ->integer('parent_id') + ->allowEmptyString('parent_id'); + + $validator + ->boolean('enabled') + ->notEmptyString('enabled'); + + return $validator; + } + + /** + * Returns a rules checker object that will be used for validating + * application integrity. + * + * @param RulesChecker $rules The rules object to be modified. + * @return RulesChecker + */ + public function buildRules(RulesChecker $rules): RulesChecker + { + $rules->add($rules->isUnique(['product_catalog_id', 'name']), ['errorField' => 'product_catalog_id']); + $rules->add($rules->existsIn(['product_catalog_id'], 'ProductCatalogs'), ['errorField' => 'product_catalog_id']); + $rules->add($rules->existsIn(['parent_id'], 'ParentProductCategories'), ['errorField' => 'parent_id']); + + return $rules; + } + + /** + * @param int $catalogId + * + * @return void + */ + public function setConfigureCatalogId(string $catalogId) + { + $this->treeCatalogId = $catalogId; + $this->behaviors()->Tree->setConfig('scope', ['product_catalog_id' => $this->treeCatalogId]); + } + + /** + * @param EntityInterface $entity + * @param array $options + * + * @return EntityInterface|false + */ + public function save(EntityInterface $entity, array $options = []): EntityInterface|false + { + $this->behaviors()->get('Tree')->setConfig([ + 'scope' => [ + 'product_catalog_id' => $entity->product_catalog_id, + ], + ]); + + return parent::save($entity, $options); + } +} diff --git a/src/Model/Table/ProductCategoryAttributeOptionsTable.php b/src/Model/Table/ProductCategoryAttributeOptionsTable.php new file mode 100644 index 0000000..6e4b0e9 --- /dev/null +++ b/src/Model/Table/ProductCategoryAttributeOptionsTable.php @@ -0,0 +1,103 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Table; + +use Cake\Datasource\EntityInterface; +use Cake\Datasource\ResultSetInterface; +use Cake\ORM\Association\BelongsTo; +use Cake\ORM\Query\SelectQuery; +use Cake\ORM\RulesChecker; +use Cake\ORM\Table; +use Cake\Validation\Validator; +use CakeProducts\Model\Entity\ProductCategoryAttributeOption; +use Closure; +use Psr\SimpleCache\CacheInterface; + +/** + * ProductCategoryAttributeOptions Model + * + * @property ProductCategoryAttributesTable&BelongsTo $ProductCategoryAttributes + * + * @method ProductCategoryAttributeOption newEmptyEntity() + * @method ProductCategoryAttributeOption newEntity(array $data, array $options = []) + * @method array<ProductCategoryAttributeOption> newEntities(array $data, array $options = []) + * @method ProductCategoryAttributeOption get(mixed $primaryKey, array|string $finder = 'all', CacheInterface|string|null $cache = null, Closure|string|null $cacheKey = null, mixed ...$args) + * @method ProductCategoryAttributeOption findOrCreate($search, ?callable $callback = null, array $options = []) + * @method ProductCategoryAttributeOption patchEntity(EntityInterface $entity, array $data, array $options = []) + * @method array<ProductCategoryAttributeOption> patchEntities(iterable $entities, array $data, array $options = []) + * @method ProductCategoryAttributeOption|false save(EntityInterface $entity, array $options = []) + * @method ProductCategoryAttributeOption saveOrFail(EntityInterface $entity, array $options = []) + * @method iterable<ProductCategoryAttributeOption>|ResultSetInterface<ProductCategoryAttributeOption>|false saveMany(iterable $entities, array $options = []) + * @method iterable<ProductCategoryAttributeOption>|ResultSetInterface<ProductCategoryAttributeOption> saveManyOrFail(iterable $entities, array $options = []) + * @method iterable<ProductCategoryAttributeOption>|ResultSetInterface<ProductCategoryAttributeOption>|false deleteMany(iterable $entities, array $options = []) + * @method iterable<ProductCategoryAttributeOption>|ResultSetInterface<ProductCategoryAttributeOption> deleteManyOrFail(iterable $entities, array $options = []) + */ +class ProductCategoryAttributeOptionsTable extends Table +{ + /** + * Initialize method + * + * @param array<string, mixed> $config The configuration for the Table. + * @return void + */ + public function initialize(array $config): void + { + parent::initialize($config); + + $this->setTable('product_category_attribute_options'); + $this->setDisplayField('attribute_value'); + $this->setPrimaryKey('id'); + + $this->belongsTo('ProductCategoryAttributes', [ + 'foreignKey' => 'product_category_attribute_id', + 'joinType' => 'INNER', + 'className' => 'CakeProducts.ProductCategoryAttributes', + ]); + } + + /** + * Default validation rules. + * + * @param Validator $validator Validator instance. + * @return Validator + */ + public function validationDefault(Validator $validator): Validator + { + $validator + ->integer('product_category_attribute_id') + ->notEmptyString('product_category_attribute_id'); + + $validator + ->scalar('attribute_value') + ->maxLength('attribute_value', 255) + ->requirePresence('attribute_value', 'create') + ->notEmptyString('attribute_value'); + + $validator + ->scalar('attribute_label') + ->maxLength('attribute_label', 255) + ->requirePresence('attribute_label', 'create') + ->notEmptyString('attribute_label'); + + $validator + ->boolean('enabled') + ->notEmptyString('enabled'); + + return $validator; + } + + /** + * Returns a rules checker object that will be used for validating + * application integrity. + * + * @param RulesChecker $rules The rules object to be modified. + * @return RulesChecker + */ + public function buildRules(RulesChecker $rules): RulesChecker + { + $rules->add($rules->existsIn(['product_category_attribute_id'], 'ProductCategoryAttributes'), ['errorField' => '0']); + + return $rules; + } +} diff --git a/src/Model/Table/ProductCategoryAttributesTable.php b/src/Model/Table/ProductCategoryAttributesTable.php new file mode 100644 index 0000000..a2f3cf8 --- /dev/null +++ b/src/Model/Table/ProductCategoryAttributesTable.php @@ -0,0 +1,113 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Table; + +use Cake\Database\Type\EnumType; +use Cake\Datasource\EntityInterface; +use Cake\Datasource\ResultSetInterface; +use Cake\ORM\Association\BelongsTo; +use Cake\ORM\Query\SelectQuery; +use Cake\ORM\RulesChecker; +use Cake\ORM\Table; +use Cake\Validation\Validator; +use CakeProducts\Model\Entity\ProductCategoryAttribute; +use CakeProducts\Model\Enum\ProductCategoryAttributeTypeId; +use Closure; +use Psr\SimpleCache\CacheInterface; + +/** + * ProductCategoryAttributes Model + * + * @property ProductCategoriesTable&BelongsTo $ProductCategories + * + * @method ProductCategoryAttribute newEmptyEntity() + * @method ProductCategoryAttribute newEntity(array $data, array $options = []) + * @method array<ProductCategoryAttribute> newEntities(array $data, array $options = []) + * @method ProductCategoryAttribute get(mixed $primaryKey, array|string $finder = 'all', CacheInterface|string|null $cache = null, Closure|string|null $cacheKey = null, mixed ...$args) + * @method ProductCategoryAttribute findOrCreate($search, ?callable $callback = null, array $options = []) + * @method ProductCategoryAttribute patchEntity(EntityInterface $entity, array $data, array $options = []) + * @method array<ProductCategoryAttribute> patchEntities(iterable $entities, array $data, array $options = []) + * @method ProductCategoryAttribute|false save(EntityInterface $entity, array $options = []) + * @method ProductCategoryAttribute saveOrFail(EntityInterface $entity, array $options = []) + * @method iterable<ProductCategoryAttribute>|ResultSetInterface<ProductCategoryAttribute>|false saveMany(iterable $entities, array $options = []) + * @method iterable<ProductCategoryAttribute>|ResultSetInterface<ProductCategoryAttribute> saveManyOrFail(iterable $entities, array $options = []) + * @method iterable<ProductCategoryAttribute>|ResultSetInterface<ProductCategoryAttribute>|false deleteMany(iterable $entities, array $options = []) + * @method iterable<ProductCategoryAttribute>|ResultSetInterface<ProductCategoryAttribute> deleteManyOrFail(iterable $entities, array $options = []) + */ +class ProductCategoryAttributesTable extends Table +{ + /** + * Initialize method + * + * @param array<string, mixed> $config The configuration for the Table. + * @return void + */ + public function initialize(array $config): void + { + parent::initialize($config); + + $this->setTable('product_category_attributes'); + $this->setDisplayField('name'); + $this->setPrimaryKey('id'); + + $this->belongsTo('ProductCategories', [ + 'foreignKey' => 'product_category_id', + 'bindingKey' => 'internal_id', + 'className' => 'CakeProducts.ProductCategories', + ]); + + $this->hasMany('ProductCategoryAttributeOptions', [ + 'foreignKey' => 'product_category_attribute_id', + 'className' => 'CakeProducts.ProductCategoryAttributeOptions', + 'saveStrategy' => 'replace', + ]); + $this->getSchema()->setColumnType('attribute_type_id', EnumType::from(ProductCategoryAttributeTypeId::class)); + } + + /** + * Default validation rules. + * + * @param Validator $validator Validator instance. + * @return Validator + */ + public function validationDefault(Validator $validator): Validator + { + $validator + ->scalar('name') + ->maxLength('name', 255) + ->requirePresence('name', 'create') + ->notEmptyString('name'); + + $validator + ->uuid('product_category_id') + ->allowEmptyString('product_category_id'); + + $validator + ->integer('attribute_type_id') + ->requirePresence('attribute_type_id', 'create') + ->notEmptyString('attribute_type_id'); + + $validator + ->boolean('enabled') + ->requirePresence('enabled', 'create') + ->notEmptyString('enabled'); + + return $validator; + } + + /** + * Returns a rules checker object that will be used for validating + * application integrity. + * + * @param RulesChecker $rules The rules object to be modified. + * @return RulesChecker + */ + public function buildRules(RulesChecker $rules): RulesChecker + { + $rules->add($rules->isUnique(['name', 'product_category_id'], ['allowMultipleNulls' => true]), ['errorField' => 'name']); + $rules->add($rules->existsIn(['product_category_id'], 'ProductCategories'), ['errorField' => 'product_category_id']); + + return $rules; + } +} diff --git a/src/Model/Table/ProductsTable.php b/src/Model/Table/ProductsTable.php new file mode 100644 index 0000000..e280630 --- /dev/null +++ b/src/Model/Table/ProductsTable.php @@ -0,0 +1,105 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Model\Table; + +use Cake\Database\Type\EnumType; +use Cake\Datasource\EntityInterface; +use Cake\Datasource\ResultSetInterface; +use Cake\ORM\Association\BelongsTo; +use Cake\ORM\Query\SelectQuery; +use Cake\ORM\RulesChecker; +use Cake\ORM\Table; +use Cake\Validation\Validator; +use CakeProducts\Model\Entity\Product; +use CakeProducts\Model\Enum\ProductProductTypeId; +use Closure; +use Psr\SimpleCache\CacheInterface; + +/** + * Products Model + * + * @property ProductCategoriesTable&BelongsTo $ProductCategories + * + * @method Product newEmptyEntity() + * @method Product newEntity(array $data, array $options = []) + * @method array<Product> newEntities(array $data, array $options = []) + * @method Product get(mixed $primaryKey, array|string $finder = 'all', CacheInterface|string|null $cache = null, Closure|string|null $cacheKey = null, mixed ...$args) + * @method Product findOrCreate($search, ?callable $callback = null, array $options = []) + * @method Product patchEntity(EntityInterface $entity, array $data, array $options = []) + * @method array<Product> patchEntities(iterable $entities, array $data, array $options = []) + * @method Product|false save(EntityInterface $entity, array $options = []) + * @method Product saveOrFail(EntityInterface $entity, array $options = []) + * @method iterable<Product>|ResultSetInterface<Product>|false saveMany(iterable $entities, array $options = []) + * @method iterable<Product>|ResultSetInterface<Product> saveManyOrFail(iterable $entities, array $options = []) + * @method iterable<Product>|ResultSetInterface<Product>|false deleteMany(iterable $entities, array $options = []) + * @method iterable<Product>|ResultSetInterface<Product> deleteManyOrFail(iterable $entities, array $options = []) + */ +class ProductsTable extends Table +{ + /** + * Initialize method + * + * @param array<string, mixed> $config The configuration for the Table. + * @return void + */ + public function initialize(array $config): void + { + parent::initialize($config); + + $this->setTable('products'); + $this->setDisplayField('name'); + $this->setPrimaryKey('id'); + + $this->belongsTo('ProductCategories', [ + 'foreignKey' => 'product_category_id', + 'bindingKey' => 'internal_id', + 'joinType' => 'INNER', + 'className' => 'CakeProducts.ProductCategories', + ]); + + $this->getSchema()->setColumnType('product_type_id', EnumType::from(ProductProductTypeId::class)); + + } + + /** + * Default validation rules. + * + * @param Validator $validator Validator instance. + * @return Validator + */ + public function validationDefault(Validator $validator): Validator + { + $validator + ->scalar('name') + ->maxLength('name', 255) + ->requirePresence('name', 'create') + ->notEmptyString('name'); + + $validator + ->uuid('product_category_id') + ->notEmptyString('product_category_id'); + + $validator + ->integer('product_type_id') + ->requirePresence('product_type_id', 'create') + ->notEmptyString('product_type_id'); + + return $validator; + } + + /** + * Returns a rules checker object that will be used for validating + * application integrity. + * + * @param RulesChecker $rules The rules object to be modified. + * @return RulesChecker + */ + public function buildRules(RulesChecker $rules): RulesChecker + { + $rules->add($rules->isUnique(['product_category_id', 'name']), ['errorField' => '0']); + $rules->add($rules->existsIn(['product_category_id'], 'ProductCategories'), ['errorField' => '1']); + + return $rules; + } +} diff --git a/src/Service/CatalogManagerServiceProvider.php b/src/Service/CatalogManagerServiceProvider.php new file mode 100644 index 0000000..c0ef7e3 --- /dev/null +++ b/src/Service/CatalogManagerServiceProvider.php @@ -0,0 +1,29 @@ +<?php + +namespace CakeProducts\Service; + +use Cake\Core\ContainerInterface; +use Cake\Core\Plugin; +use Cake\Core\ServiceConfig; +use Cake\Core\ServiceProvider; +use Cake\Log\Log; +use Cake\ORM\Locator\LocatorAwareTrait; +use CakeProducts\Service\InternalCatalogManagerService; + +class CatalogManagerServiceProvider extends ServiceProvider +{ + protected array $provides = [ + InternalCatalogManagerService::class + ]; + + /** + * @param ContainerInterface $container + * + * @return void + */ + public function services(ContainerInterface $container): void + { + $container->add(InternalCatalogManagerService::class) + ->addArgument(new ExternalCatalogManagerService()); + } +} diff --git a/src/Service/ExternalCatalogManagerService.php b/src/Service/ExternalCatalogManagerService.php new file mode 100644 index 0000000..336bf86 --- /dev/null +++ b/src/Service/ExternalCatalogManagerService.php @@ -0,0 +1,184 @@ +<?php + +namespace CakeProducts\Service; + +use Cake\Cache\Cache; +use Cake\Core\ServiceConfig; +use Cake\Http\Client; +use Cake\I18n\FrozenTime; +use Cake\I18n\Time; +use Cake\Log\Log; +use Cake\ORM\Locator\LocatorAwareTrait; +use CakeProducts\Model\Entity\ExternalProductCatalog; +use CakeProducts\Model\Entity\ProductCategory; +use CakeProducts\Model\Table\ProductCatalogsTable; + +class ExternalCatalogManagerService +{ + use LocatorAwareTrait; + + /** + * @var ProductCatalogsTable + */ + protected \Cake\ORM\Table|ProductCatalogsTable $ProductCatalogs; + + /** + * @var ServiceConfig + */ + protected ServiceConfig $serviceConfig; + + /** + * @var Client + */ + protected Client $httpClient; + + /** + * + */ + public function __construct() + { + $this->ProductCatalogs = $this->fetchTable('CakeProducts.ProductCatalogs'); + $this->serviceConfig = new ServiceConfig(); + $this->httpClient = new Client([ +// 'host' => $config['base_url'], +// 'scheme' => 'https', +// 'scheme' => 'http', + ]); + + } + + public function newCategoryCreated(ProductCategory $productCategory) + { + $results = []; + + $externalProductCatalogs = $this->_getExternalProductCatalogsForCatalogId($productCategory->product_catalog_id); + foreach ($externalProductCatalogs as $externalProductCatalog) { + $results[] = $this->_createNewCategoryForExternalProductCatalog($externalProductCatalog, $productCategory); + } + + return $results; + } + + protected function _createNewCategoryForExternalProductCatalog(ExternalProductCatalog $externalProductCatalog, ProductCategory $productCategory) + { + $url = $externalProductCatalog->api_url . '/product-categories'; + $response = $this->postToUrl($url, $productCategory->toArray()); + + Log::debug(print_r('$response->getJson()', true)); + Log::debug(print_r($response->getJson(), true)); + Log::debug(print_r('$response->getStatusCode()', true)); + Log::debug(print_r($response->getStatusCode(), true)); + + return $response->getStatusCode(); + } + + /** + * @return mixed|null + */ + public function getJwtToken() + { + Log::debug('inside getJwtToken'); + if (Cache::read('product_catalog_api_token')) { + Log::debug('token was cached'); +// return Cache::read('product_catalog_api_token'); + } else { + Log::debug('token was **NOT** cached'); + } + + $response = $this->httpClient->post('http://localhost:8766/api/v1/users/token', json_encode([ + 'username' => 'test', + 'password' => 'test', + ]), ['headers' => ['Accept' => 'application/json', 'Content-Type' => 'application/json']]); +// $this->httpClient->getConfig(); + if ($response->isOk()) { + $json = $response->getJson(); + $token = array_key_exists('token', $json) ? $json['token'] : null; + Cache::write('product_catalog_api_token', $token); + Log::debug('$token'); + Log::debug($token); + + return $token; + } + Log::debug('$response->getStringBody()'); + Log::debug($response->getStringBody()); + Log::debug(print_r('$response->getStatusCode()', true)); + Log::debug(print_r($response->getStatusCode(), true)); + + return null; + } + + public function postToUrl(string $url, array $data, int $tries = 0) + { +// if (true || !Cache::read('product_catalog_api_token')) { + $token = $this->getJwtToken(); +// } + Log::debug('$token inside postToUrl' . $token); + + Log::debug('Cache::read(product_catalog_api_token)'); + Log::debug(Cache::read('product_catalog_api_token') ? Cache::read('product_catalog_api_token') : 'NULL'); + Log::debug('ATTEMPT # ' . $tries); + + $response = $this->httpClient->post($url, json_encode($data), [ + 'headers' => [ + 'Accept' => 'application/json', + 'Content-Type' => 'application/json', +// 'Authorization' => 'Bearer ' . base64_encode(Cache::read('product_catalog_api_token')) + 'Authorization' => 'Bearer ' . $token, + ] + ]); + + if (!$response->isOk()) { + $tries++; + } + if ($tries > 3) { + return $response; + } + if ($response->getStatusCode() == 401) { + $this->postToUrl($url, $data, $tries); + } + Log::debug('$response->getJson'); + Log::debug(print_r($response->getJson(), true)); + + return $response; + } + + /** + * @param string $url + * @param array $data + * @param int $tries + * + * @return mixed + */ + public function putToUrl(string $url, array $data, int $tries = 0) + { + if (!Cache::read('product_catalog_api_token')) { + $this->getJwtToken(); + } + + $response = $this->httpClient->put($url, json_encode($data), [ + 'headers' => [ + 'Accept' => 'application/json', + 'Content-Type' => 'application/json', + 'Authorization' => 'Bearer ' . Cache::read('product_catalog_api_token') + ], + ]); + + if (!$response->isOk()) { + $tries++; + } + if ($tries > 3) { + return $response; + } + if ($response->getStatusCode() == 401) { + $this->getJwtToken(); + $this->putToUrl($url, $data, $tries); + } + + return $response; + } + + protected function _getExternalProductCatalogsForCatalogId(string $productCatalogId) + { + return $this->ProductCatalogs->ExternalProductCatalogs->find()->where(['product_catalog_id' => $productCatalogId])->toArray(); + } +} diff --git a/src/Service/InternalCatalogManagerService.php b/src/Service/InternalCatalogManagerService.php new file mode 100644 index 0000000..28aaa94 --- /dev/null +++ b/src/Service/InternalCatalogManagerService.php @@ -0,0 +1,106 @@ +<?php + +namespace CakeProducts\Service; + +use Cake\Core\ServiceConfig; +use Cake\Datasource\EntityInterface; +use Cake\I18n\FrozenTime; +use Cake\I18n\Time; +use Cake\Log\Log; +use Cake\ORM\Locator\LocatorAwareTrait; +use Cake\ORM\Table; +use Cake\Utility\Text; +use CakeProducts\Model\Entity\ProductCategory; +use CakeProducts\Model\Table\ProductCatalogsTable; + +class InternalCatalogManagerService +{ + use LocatorAwareTrait; + + /** + * @var ProductCatalogsTable + */ + protected Table|ProductCatalogsTable $ProductCatalogs; + + /** + * @var ServiceConfig + */ + protected ServiceConfig $serviceConfig; + + /** + * @var ExternalCatalogManagerService|null + */ + protected ?ExternalCatalogManagerService $externalCatalogManager; + + /** + * + */ + public function __construct(ExternalCatalogManagerService $externalCatalogManagerService) + { + $this->ProductCatalogs = $this->fetchTable('CakeProducts.ProductCatalogs'); + $this->serviceConfig = new ServiceConfig(); + + if ($this->serviceConfig->get('CakeProducts.internal.enabled') && $this->serviceConfig->get('CakeProducts.internal.syncExternally')) { + $this->externalCatalogManager = $externalCatalogManagerService; + } + + } + public function getCatalog(string $id = null) + { + $contain = ['ProductCategories']; + if ($this->serviceConfig->get('CakeProducts.internal.syncExternally')) { + $contain[] = 'ExternalProductCatalogs'; + } + + return $this->ProductCatalogs->get($id, contain: $contain); + } + + /** + * @param string|null $id + * + * @return \App\Model\Entity\ProductCategory|EntityInterface + */ + public function getCategory(string $id = null) + { + $contain = ['ProductCatalogs', 'ParentProductCategories', 'ChildProductCategories']; + + return $this->ProductCatalogs->ProductCategories->get($id, contain: $contain); + } + + /** + * @param ProductCategory $productCategory product category entity + * @param array $data data to save + * + * @return array + */ + public function createNewCategory(ProductCategory $productCategory, array $data = []): array + { + $now = Time::now(); + $associated = []; + + Log::info('posted data - adding new ProductCategory'); + Log::info(print_r($data, true)); + + $saveOptions = [ + 'associated' => $associated, + ]; + if (!array_key_exists('internal_id', $data) || !$data['internal_id']) { + $data['internal_id'] = Text::uuid(); + } + $productCategory = $this->ProductCatalogs->ProductCategories->patchEntity($productCategory, $data, $saveOptions); + if ($productCategory->getErrors()) { + Log::debug(print_r('$productCategory->getErrors() next - failed to save from create new product category', true)); + Log::debug(print_r($productCategory->getErrors(), true)); + } + $returnData = [ + 'entity' => $productCategory, + 'result' => $this->ProductCatalogs->ProductCategories->save($productCategory, $saveOptions), + 'apiResults' => [], + ]; + if ($returnData['result'] && $this->externalCatalogManager) { + $returnData['apiResults'] = $this->externalCatalogManager->newCategoryCreated($returnData['result']); + } + + return $returnData; + } +} diff --git a/templates/ExternalProductCatalogs/add.php b/templates/ExternalProductCatalogs/add.php new file mode 100644 index 0000000..1c06841 --- /dev/null +++ b/templates/ExternalProductCatalogs/add.php @@ -0,0 +1,32 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $externalProductCatalog + * @var \Cake\Collection\CollectionInterface|string[] $productCatalogs + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('List External Product Catalogs'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="externalProductCatalogs form content"> + <?= $this->Form->create($externalProductCatalog) ?> + <fieldset> + <legend><?= __('Add External Product Catalog') ?></legend> + <?php + echo $this->Form->control('product_catalog_id', ['options' => $productCatalogs]); + echo $this->Form->control('base_url'); + echo $this->Form->control('api_url'); + echo $this->Form->control('deleted', ['empty' => true]); + echo $this->Form->control('enabled'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/ExternalProductCatalogs/edit.php b/templates/ExternalProductCatalogs/edit.php new file mode 100644 index 0000000..3485c52 --- /dev/null +++ b/templates/ExternalProductCatalogs/edit.php @@ -0,0 +1,37 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $externalProductCatalog + * @var string[]|\Cake\Collection\CollectionInterface $productCatalogs + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Form->postLink( + __('Delete'), + ['action' => 'delete', $externalProductCatalog->id], + ['confirm' => __('Are you sure you want to delete # {0}?', $externalProductCatalog->id), 'class' => 'side-nav-item'] + ) ?> + <?= $this->Html->link(__('List External Product Catalogs'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="externalProductCatalogs form content"> + <?= $this->Form->create($externalProductCatalog) ?> + <fieldset> + <legend><?= __('Edit External Product Catalog') ?></legend> + <?php + echo $this->Form->control('product_catalog_id', ['options' => $productCatalogs]); + echo $this->Form->control('base_url'); + echo $this->Form->control('api_url'); + echo $this->Form->control('deleted', ['empty' => true]); + echo $this->Form->control('enabled'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/ExternalProductCatalogs/index.php b/templates/ExternalProductCatalogs/index.php new file mode 100644 index 0000000..eca326b --- /dev/null +++ b/templates/ExternalProductCatalogs/index.php @@ -0,0 +1,54 @@ +<?php +/** + * @var \App\View\AppView $this + * @var iterable<\Cake\Datasource\EntityInterface> $externalProductCatalogs + */ +?> +<div class="externalProductCatalogs index content"> + <?= $this->Html->link(__('New External Product Catalog'), ['action' => 'add'], ['class' => 'button float-right']) ?> + <h3><?= __('External Product Catalogs') ?></h3> + <div class="table-responsive"> + <table> + <thead> + <tr> + <th><?= $this->Paginator->sort('id') ?></th> + <th><?= $this->Paginator->sort('product_catalog_id') ?></th> + <th><?= $this->Paginator->sort('base_url') ?></th> + <th><?= $this->Paginator->sort('api_url') ?></th> + <th><?= $this->Paginator->sort('created') ?></th> + <th><?= $this->Paginator->sort('deleted') ?></th> + <th><?= $this->Paginator->sort('enabled') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + </thead> + <tbody> + <?php foreach ($externalProductCatalogs as $externalProductCatalog): ?> + <tr> + <td><?= $this->Number->format($externalProductCatalog->id) ?></td> + <td><?= $externalProductCatalog->hasValue('product_catalog') ? $this->Html->link($externalProductCatalog->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $externalProductCatalog->product_catalog->id]) : '' ?></td> + <td><?= h($externalProductCatalog->base_url) ?></td> + <td><?= h($externalProductCatalog->api_url) ?></td> + <td><?= h($externalProductCatalog->created) ?></td> + <td><?= h($externalProductCatalog->deleted) ?></td> + <td><?= h($externalProductCatalog->enabled) ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['action' => 'view', $externalProductCatalog->id]) ?> + <?= $this->Html->link(__('Edit'), ['action' => 'edit', $externalProductCatalog->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['action' => 'delete', $externalProductCatalog->id], ['confirm' => __('Are you sure you want to delete # {0}?', $externalProductCatalog->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </tbody> + </table> + </div> + <div class="paginator"> + <ul class="pagination"> + <?= $this->Paginator->first('<< ' . __('first')) ?> + <?= $this->Paginator->prev('< ' . __('previous')) ?> + <?= $this->Paginator->numbers() ?> + <?= $this->Paginator->next(__('next') . ' >') ?> + <?= $this->Paginator->last(__('last') . ' >>') ?> + </ul> + <p><?= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?></p> + </div> +</div> diff --git a/templates/ExternalProductCatalogs/view.php b/templates/ExternalProductCatalogs/view.php new file mode 100644 index 0000000..abe3e59 --- /dev/null +++ b/templates/ExternalProductCatalogs/view.php @@ -0,0 +1,52 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $externalProductCatalog + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('Edit External Product Catalog'), ['action' => 'edit', $externalProductCatalog->id], ['class' => 'side-nav-item']) ?> + <?= $this->Form->postLink(__('Delete External Product Catalog'), ['action' => 'delete', $externalProductCatalog->id], ['confirm' => __('Are you sure you want to delete # {0}?', $externalProductCatalog->id), 'class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('List External Product Catalogs'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('New External Product Catalog'), ['action' => 'add'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="externalProductCatalogs view content"> + <h3><?= h($externalProductCatalog->base_url) ?></h3> + <table> + <tr> + <th><?= __('Product Catalog') ?></th> + <td><?= $externalProductCatalog->hasValue('product_catalog') ? $this->Html->link($externalProductCatalog->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $externalProductCatalog->product_catalog->id]) : '' ?></td> + </tr> + <tr> + <th><?= __('Base Url') ?></th> + <td><?= h($externalProductCatalog->base_url) ?></td> + </tr> + <tr> + <th><?= __('Api Url') ?></th> + <td><?= h($externalProductCatalog->api_url) ?></td> + </tr> + <tr> + <th><?= __('Id') ?></th> + <td><?= $this->Number->format($externalProductCatalog->id) ?></td> + </tr> + <tr> + <th><?= __('Created') ?></th> + <td><?= h($externalProductCatalog->created) ?></td> + </tr> + <tr> + <th><?= __('Deleted') ?></th> + <td><?= h($externalProductCatalog->deleted) ?></td> + </tr> + <tr> + <th><?= __('Enabled') ?></th> + <td><?= $externalProductCatalog->enabled ? __('Yes') : __('No'); ?></td> + </tr> + </table> + </div> + </div> +</div> diff --git a/templates/ProductCatalogs/add.php b/templates/ProductCatalogs/add.php new file mode 100644 index 0000000..af1d12f --- /dev/null +++ b/templates/ProductCatalogs/add.php @@ -0,0 +1,29 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCatalog + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('List Product Catalogs'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCatalogs form content"> + <?= $this->Form->create($productCatalog) ?> + <fieldset> + <legend><?= __('Add Product Catalog') ?></legend> + <?php + echo $this->Form->control('name'); + echo $this->Form->control('catalog_description'); + echo $this->Form->control('enabled'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/ProductCatalogs/edit.php b/templates/ProductCatalogs/edit.php new file mode 100644 index 0000000..5eecae1 --- /dev/null +++ b/templates/ProductCatalogs/edit.php @@ -0,0 +1,34 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCatalog + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Form->postLink( + __('Delete'), + ['action' => 'delete', $productCatalog->id], + ['confirm' => __('Are you sure you want to delete # {0}?', $productCatalog->id), 'class' => 'side-nav-item'] + ) ?> + <?= $this->Html->link(__('List Product Catalogs'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCatalogs form content"> + <?= $this->Form->create($productCatalog) ?> + <fieldset> + <legend><?= __('Edit Product Catalog') ?></legend> + <?php + echo $this->Form->control('name'); + echo $this->Form->control('catalog_description'); + echo $this->Form->control('enabled'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/ProductCatalogs/index.php b/templates/ProductCatalogs/index.php new file mode 100644 index 0000000..5dc9d2e --- /dev/null +++ b/templates/ProductCatalogs/index.php @@ -0,0 +1,48 @@ +<?php +/** + * @var \App\View\AppView $this + * @var iterable<\Cake\Datasource\EntityInterface> $productCatalogs + */ +?> +<div class="productCatalogs index content"> + <?= $this->Html->link(__('New Product Catalog'), ['action' => 'add'], ['class' => 'button float-right']) ?> + <h3><?= __('Product Catalogs') ?></h3> + <div class="table-responsive"> + <table> + <thead> + <tr> + <th><?= $this->Paginator->sort('id') ?></th> + <th><?= $this->Paginator->sort('name') ?></th> + <th><?= $this->Paginator->sort('catalog_description') ?></th> + <th><?= $this->Paginator->sort('enabled') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + </thead> + <tbody> + <?php foreach ($productCatalogs as $productCatalog): ?> + <tr> + <td><?= $productCatalog->id; ?></td> + <td><?= h($productCatalog->name) ?></td> + <td><?= h($productCatalog->catalog_description) ?></td> + <td><?= h($productCatalog->enabled) ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['action' => 'view', $productCatalog->id]) ?> + <?= $this->Html->link(__('Edit'), ['action' => 'edit', $productCatalog->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['action' => 'delete', $productCatalog->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCatalog->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </tbody> + </table> + </div> + <div class="paginator"> + <ul class="pagination"> + <?= $this->Paginator->first('<< ' . __('first')) ?> + <?= $this->Paginator->prev('< ' . __('previous')) ?> + <?= $this->Paginator->numbers() ?> + <?= $this->Paginator->next(__('next') . ' >') ?> + <?= $this->Paginator->last(__('last') . ' >>') ?> + </ul> + <p><?= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?></p> + </div> +</div> diff --git a/templates/ProductCatalogs/view.php b/templates/ProductCatalogs/view.php new file mode 100644 index 0000000..d0b294c --- /dev/null +++ b/templates/ProductCatalogs/view.php @@ -0,0 +1,73 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCatalog + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('Edit Product Catalog'), ['action' => 'edit', $productCatalog->id], ['class' => 'side-nav-item']) ?> + <?= $this->Form->postLink(__('Delete Product Catalog'), ['action' => 'delete', $productCatalog->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCatalog->id), 'class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('List Product Catalogs'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('New Product Catalog'), ['action' => 'add'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCatalogs view content"> + <h3><?= h($productCatalog->name) ?></h3> + <table> + <tr> + <th><?= __('Name') ?></th> + <td><?= h($productCatalog->name) ?></td> + </tr> + <tr> + <th><?= __('Catalog Description') ?></th> + <td><?= h($productCatalog->catalog_description) ?></td> + </tr> + <tr> + <th><?= __('Id') ?></th> + <td><?= $productCatalog->id; ?></td> + </tr> + <tr> + <th><?= __('Enabled') ?></th> + <td><?= $productCatalog->enabled ? __('Yes') : __('No'); ?></td> + </tr> + </table> + <div class="related"> + <h4><?= __('Related Product Categories') ?></h4> + <?php if (!empty($productCatalog->product_categories)) : ?> + <div class="table-responsive"> + <table> + <tr> + <th><?= __('Id') ?></th> + <th><?= __('Product Catalog Id') ?></th> + <th><?= __('Name') ?></th> + <th><?= __('Category Description') ?></th> + <th><?= __('Parent Id') ?></th> + <th><?= __('Enabled') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + <?php foreach ($productCatalog->product_categories as $productCategories) : ?> + <tr> + <td><?= h($productCategories->id) ?></td> + <td><?= h($productCategories->product_catalog_id) ?></td> + <td><?= h($productCategories->name) ?></td> + <td><?= h($productCategories->category_description) ?></td> + <td><?= h($productCategories->parent_id) ?></td> + <td><?= h($productCategories->enabled) ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['controller' => 'ProductCategories', 'action' => 'view', $productCategories->id]) ?> + <?= $this->Html->link(__('Edit'), ['controller' => 'ProductCategories', 'action' => 'edit', $productCategories->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['controller' => 'ProductCategories', 'action' => 'delete', $productCategories->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCategories->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </table> + </div> + <?php endif; ?> + </div> + </div> + </div> +</div> diff --git a/templates/ProductCategories/add.php b/templates/ProductCategories/add.php new file mode 100644 index 0000000..293abab --- /dev/null +++ b/templates/ProductCategories/add.php @@ -0,0 +1,33 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategory + * @var \Cake\Collection\CollectionInterface|string[] $productCatalogs + * @var \Cake\Collection\CollectionInterface|string[] $parentProductCategories + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('List Product Categories'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCategories form content"> + <?= $this->Form->create($productCategory) ?> + <fieldset> + <legend><?= __('Add Product Category') ?></legend> + <?php + echo $this->Form->control('product_catalog_id', ['options' => $productCatalogs]); + echo $this->Form->control('name'); + echo $this->Form->control('category_description'); + echo $this->Form->control('parent_id', ['options' => $parentProductCategories, 'empty' => true]); + echo $this->Form->control('enabled'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/ProductCategories/edit.php b/templates/ProductCategories/edit.php new file mode 100644 index 0000000..d2b8a22 --- /dev/null +++ b/templates/ProductCategories/edit.php @@ -0,0 +1,38 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategory + * @var string[]|\Cake\Collection\CollectionInterface $productCatalogs + * @var string[]|\Cake\Collection\CollectionInterface $parentProductCategories + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Form->postLink( + __('Delete'), + ['action' => 'delete', $productCategory->id], + ['confirm' => __('Are you sure you want to delete # {0}?', $productCategory->id), 'class' => 'side-nav-item'] + ) ?> + <?= $this->Html->link(__('List Product Categories'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCategories form content"> + <?= $this->Form->create($productCategory) ?> + <fieldset> + <legend><?= __('Edit Product Category') ?></legend> + <?php + echo $this->Form->control('product_catalog_id', ['options' => $productCatalogs]); + echo $this->Form->control('name'); + echo $this->Form->control('category_description'); + echo $this->Form->control('parent_id', ['options' => $parentProductCategories, 'empty' => true]); + echo $this->Form->control('enabled'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/ProductCategories/index.php b/templates/ProductCategories/index.php new file mode 100644 index 0000000..de301d4 --- /dev/null +++ b/templates/ProductCategories/index.php @@ -0,0 +1,50 @@ +<?php +/** + * @var \App\View\AppView $this + * @var iterable<\Cake\Datasource\EntityInterface> $productCategories + */ +?> +<div class="productCategories index content"> + <?= $this->Html->link(__('New Product Category'), ['action' => 'add'], ['class' => 'button float-right']) ?> + <h3><?= __('Product Categories') ?></h3> + <div class="table-responsive"> + <table> + <thead> + <tr> + <th><?= $this->Paginator->sort('id') ?></th> + <th><?= $this->Paginator->sort('product_catalog_id') ?></th> + <th><?= $this->Paginator->sort('name') ?></th> + <th><?= $this->Paginator->sort('parent_id') ?></th> + <th><?= $this->Paginator->sort('enabled') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + </thead> + <tbody> + <?php foreach ($productCategories as $productCategory): ?> + <tr> + <td><?= $productCategory->id; ?></td> + <td><?= $productCategory->hasValue('product_catalog') ? $this->Html->link($productCategory->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $productCategory->product_catalog->id]) : '' ?></td> + <td><?= h($productCategory->name) ?></td> + <td><?= $productCategory->hasValue('parent_product_category') ? $this->Html->link($productCategory->parent_product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategory->parent_product_category->id]) : '' ?></td> + <td><?= h($productCategory->enabled) ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['action' => 'view', $productCategory->id]) ?> + <?= $this->Html->link(__('Edit'), ['action' => 'edit', $productCategory->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['action' => 'delete', $productCategory->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCategory->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </tbody> + </table> + </div> + <div class="paginator"> + <ul class="pagination"> + <?= $this->Paginator->first('<< ' . __('first')) ?> + <?= $this->Paginator->prev('< ' . __('previous')) ?> + <?= $this->Paginator->numbers() ?> + <?= $this->Paginator->next(__('next') . ' >') ?> + <?= $this->Paginator->last(__('last') . ' >>') ?> + </ul> + <p><?= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?></p> + </div> +</div> diff --git a/templates/ProductCategories/view.php b/templates/ProductCategories/view.php new file mode 100644 index 0000000..74d8984 --- /dev/null +++ b/templates/ProductCategories/view.php @@ -0,0 +1,95 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategory + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('Edit Product Category'), ['action' => 'edit', $productCategory->id], ['class' => 'side-nav-item']) ?> + <?= $this->Form->postLink(__('Delete Product Category'), ['action' => 'delete', $productCategory->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCategory->id), 'class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('List Product Categories'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('New Product Category'), ['action' => 'add'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCategories view content"> + <h3><?= h($productCategory->name) ?></h3> + <table> + <tr> + <th><?= __('Product Catalog') ?></th> + <td><?= $productCategory->hasValue('product_catalog') ? $this->Html->link($productCategory->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $productCategory->product_catalog->id]) : '' ?></td> + </tr> + <tr> + <th><?= __('Name') ?></th> + <td><?= h($productCategory->name) ?></td> + </tr> + <tr> + <th><?= __('Parent Product Category') ?></th> + <td><?= $productCategory->hasValue('parent_product_category') ? $this->Html->link($productCategory->parent_product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategory->parent_product_category->id]) : '' ?></td> + </tr> + <tr> + <th><?= __('Id') ?></th> + <td><?= $productCategory->id; ?></td> + </tr> + <tr> + <th><?= __('Lft') ?></th> + <td><?= $this->Number->format($productCategory->lft) ?></td> + </tr> + <tr> + <th><?= __('Rght') ?></th> + <td><?= $this->Number->format($productCategory->rght) ?></td> + </tr> + <tr> + <th><?= __('Enabled') ?></th> + <td><?= $productCategory->enabled ? __('Yes') : __('No'); ?></td> + </tr> + </table> + <div class="text"> + <strong><?= __('Category Description') ?></strong> + <blockquote> + <?= $this->Text->autoParagraph(h($productCategory->category_description)); ?> + </blockquote> + </div> + <div class="related"> + <h4><?= __('Related Product Categories') ?></h4> + <?php if (!empty($productCategory->child_product_categories)) : ?> + <div class="table-responsive"> + <table> + <tr> + <th><?= __('Id') ?></th> + <th><?= __('Product Catalog Id') ?></th> + <th><?= __('Name') ?></th> + <th><?= __('Category Description') ?></th> + <th><?= __('Parent Id') ?></th> + <th><?= __('Lft') ?></th> + <th><?= __('Rght') ?></th> + <th><?= __('Enabled') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + <?php foreach ($productCategory->child_product_categories as $childProductCategories) : ?> + <tr> + <td><?= h($childProductCategories->id) ?></td> + <td><?= h($childProductCategories->product_catalog_id) ?></td> + <td><?= h($childProductCategories->name) ?></td> + <td><?= h($childProductCategories->category_description) ?></td> + <td><?= h($childProductCategories->parent_id) ?></td> + <td><?= h($childProductCategories->lft) ?></td> + <td><?= h($childProductCategories->rght) ?></td> + <td><?= h($childProductCategories->enabled) ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['controller' => 'ProductCategories', 'action' => 'view', $childProductCategories->id]) ?> + <?= $this->Html->link(__('Edit'), ['controller' => 'ProductCategories', 'action' => 'edit', $childProductCategories->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['controller' => 'ProductCategories', 'action' => 'delete', $childProductCategories->id], ['confirm' => __('Are you sure you want to delete # {0}?', $childProductCategories->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </table> + </div> + <?php endif; ?> + </div> + </div> + </div> +</div> diff --git a/templates/ProductCategoryAttributeOptions/add.php b/templates/ProductCategoryAttributeOptions/add.php new file mode 100644 index 0000000..9025b01 --- /dev/null +++ b/templates/ProductCategoryAttributeOptions/add.php @@ -0,0 +1,20 @@ +<?php + +use App\View\AppView; +use Cake\Datasource\EntityInterface; + +/** + * @var AppView $this + * @var EntityInterface $productCategoryAttributeOption + */ + +$this->setLayout('ajax'); +$prefix = $prefix ?? ''; +if ($this->request->getQuery('prefix') !== null) { + $prefix = 'product_category_attribute_options.' . $this->request->getQuery('prefix') . '.'; +} +echo '<hr class="my-2">'; +echo $this->element('ProductCategoryAttributes/product_category_attribute_option_form', [ + 'prefix' => $prefix +]); +?> diff --git a/templates/ProductCategoryAttributes/add.php b/templates/ProductCategoryAttributes/add.php new file mode 100644 index 0000000..877275f --- /dev/null +++ b/templates/ProductCategoryAttributes/add.php @@ -0,0 +1,27 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategoryAttribute + * @var \Cake\Collection\CollectionInterface|string[] $productCategories + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('List Product Category Attributes'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCategoryAttributes form content"> + <?= $this->Form->create($productCategoryAttribute) ?> + <fieldset> + <legend><?= __('Add Product Category Attribute') ?></legend> + <?= $this->element('ProductCategoryAttributes/form'); ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> +<?= $this->Html->script('CakeProducts.product_category_attribute_options.js'); ?> diff --git a/templates/ProductCategoryAttributes/edit.php b/templates/ProductCategoryAttributes/edit.php new file mode 100644 index 0000000..f1e2ad1 --- /dev/null +++ b/templates/ProductCategoryAttributes/edit.php @@ -0,0 +1,32 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategoryAttribute + * @var string[]|\Cake\Collection\CollectionInterface $productCategories + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Form->postLink( + __('Delete'), + ['action' => 'delete', $productCategoryAttribute->id], + ['confirm' => __('Are you sure you want to delete # {0}?', $productCategoryAttribute->id), 'class' => 'side-nav-item'] + ) ?> + <?= $this->Html->link(__('List Product Category Attributes'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCategoryAttributes form content"> + <?= $this->Form->create($productCategoryAttribute) ?> + <fieldset> + <legend><?= __('Edit Product Category Attribute') ?></legend> + <?= $this->element('ProductCategoryAttributes/form'); ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> +<?= $this->Html->script('CakeProducts.product_category_attribute_options.js'); ?> diff --git a/templates/ProductCategoryAttributes/index.php b/templates/ProductCategoryAttributes/index.php new file mode 100644 index 0000000..cf164c5 --- /dev/null +++ b/templates/ProductCategoryAttributes/index.php @@ -0,0 +1,50 @@ +<?php +/** + * @var \App\View\AppView $this + * @var iterable<\Cake\Datasource\EntityInterface> $productCategoryAttributes + */ +?> +<div class="productCategoryAttributes index content"> + <?= $this->Html->link(__('New Product Category Attribute'), ['action' => 'add'], ['class' => 'button float-right']) ?> + <h3><?= __('Product Category Attributes') ?></h3> + <div class="table-responsive"> + <table> + <thead> + <tr> + <th><?= $this->Paginator->sort('id') ?></th> + <th><?= $this->Paginator->sort('name') ?></th> + <th><?= $this->Paginator->sort('product_category_id') ?></th> + <th><?= $this->Paginator->sort('attribute_type_id') ?></th> + <th><?= $this->Paginator->sort('enabled') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + </thead> + <tbody> + <?php foreach ($productCategoryAttributes as $productCategoryAttribute): ?> + <tr> + <td><?= $productCategoryAttribute->id ?></td> + <td><?= h($productCategoryAttribute->name) ?></td> + <td><?= $productCategoryAttribute->hasValue('product_category') ? $this->Html->link($productCategoryAttribute->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategoryAttribute->product_category->id]) : '' ?></td> + <td><?= $productCategoryAttribute->attribute_type_id->name ?></td> + <td><?= h($productCategoryAttribute->enabled) ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['action' => 'view', $productCategoryAttribute->id]) ?> + <?= $this->Html->link(__('Edit'), ['action' => 'edit', $productCategoryAttribute->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['action' => 'delete', $productCategoryAttribute->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCategoryAttribute->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </tbody> + </table> + </div> + <div class="paginator"> + <ul class="pagination"> + <?= $this->Paginator->first('<< ' . __('first')) ?> + <?= $this->Paginator->prev('< ' . __('previous')) ?> + <?= $this->Paginator->numbers() ?> + <?= $this->Paginator->next(__('next') . ' >') ?> + <?= $this->Paginator->last(__('last') . ' >>') ?> + </ul> + <p><?= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?></p> + </div> +</div> diff --git a/templates/ProductCategoryAttributes/view.php b/templates/ProductCategoryAttributes/view.php new file mode 100644 index 0000000..08d57ec --- /dev/null +++ b/templates/ProductCategoryAttributes/view.php @@ -0,0 +1,71 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategoryAttribute + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('Edit Product Category Attribute'), ['action' => 'edit', $productCategoryAttribute->id], ['class' => 'side-nav-item']) ?> + <?= $this->Form->postLink(__('Delete Product Category Attribute'), ['action' => 'delete', $productCategoryAttribute->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCategoryAttribute->id), 'class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('List Product Category Attributes'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('New Product Category Attribute'), ['action' => 'add'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="productCategoryAttributes view content"> + <h3><?= h($productCategoryAttribute->name) ?></h3> + <table> + <tr> + <th><?= __('Name') ?></th> + <td><?= h($productCategoryAttribute->name) ?></td> + </tr> + <tr> + <th><?= __('Product Category') ?></th> + <td><?= $productCategoryAttribute->hasValue('product_category') ? $this->Html->link($productCategoryAttribute->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategoryAttribute->product_category->id]) : '' ?></td> + </tr> + <tr> + <th><?= __('Id') ?></th> + <td><?= $productCategoryAttribute->id ?></td> + </tr> + <tr> + <th><?= __('Enabled') ?></th> + <td><?= $productCategoryAttribute->enabled ? __('Yes') : __('No'); ?></td> + </tr> + </table> + <div class="related"> + <h4><?= __('Related Product Category Attribute Options') ?></h4> + <?php if (!empty($productCategoryAttribute->product_category_attribute_options)) : ?> + <div class="table-responsive"> + <table> + <tr> + <th><?= __('Id') ?></th> + <th><?= __('Product Category Attribute Id') ?></th> + <th><?= __('Attribute Value') ?></th> + <th><?= __('Attribute Label') ?></th> + <th><?= __('Enabled') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + <?php foreach ($productCategoryAttribute->product_category_attribute_options as $productCategoryAttributeOptions) : ?> + <tr> + <td><?= h($productCategoryAttributeOptions->id) ?></td> + <td><?= h($productCategoryAttributeOptions->product_category_attribute_id) ?></td> + <td><?= h($productCategoryAttributeOptions->attribute_value) ?></td> + <td><?= h($productCategoryAttributeOptions->attribute_label) ?></td> + <td><?= h($productCategoryAttributeOptions->enabled) ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['controller' => 'ProductCategoryAttributeOptions', 'action' => 'view', $productCategoryAttributeOptions->id]) ?> + <?= $this->Html->link(__('Edit'), ['controller' => 'ProductCategoryAttributeOptions', 'action' => 'edit', $productCategoryAttributeOptions->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['controller' => 'ProductCategoryAttributeOptions', 'action' => 'delete', $productCategoryAttributeOptions->id], ['confirm' => __('Are you sure you want to delete # {0}?', $productCategoryAttributeOptions->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </table> + </div> + <?php endif; ?> + </div> + </div> + </div> +</div> diff --git a/templates/Products/add.php b/templates/Products/add.php new file mode 100644 index 0000000..39bf6b0 --- /dev/null +++ b/templates/Products/add.php @@ -0,0 +1,30 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $product + * @var \Cake\Collection\CollectionInterface|string[] $productCategories + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('List Products'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="products form content"> + <?= $this->Form->create($product) ?> + <fieldset> + <legend><?= __('Add Product') ?></legend> + <?php + echo $this->Form->control('name'); + echo $this->Form->control('product_category_id', ['options' => $productCategories]); + echo $this->Form->control('product_type_id'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/Products/edit.php b/templates/Products/edit.php new file mode 100644 index 0000000..5d49a17 --- /dev/null +++ b/templates/Products/edit.php @@ -0,0 +1,35 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $product + * @var string[]|\Cake\Collection\CollectionInterface $productCategories + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Form->postLink( + __('Delete'), + ['action' => 'delete', $product->id], + ['confirm' => __('Are you sure you want to delete # {0}?', $product->id), 'class' => 'side-nav-item'] + ) ?> + <?= $this->Html->link(__('List Products'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="products form content"> + <?= $this->Form->create($product) ?> + <fieldset> + <legend><?= __('Edit Product') ?></legend> + <?php + echo $this->Form->control('name'); + echo $this->Form->control('product_category_id', ['options' => $productCategories]); + echo $this->Form->control('product_type_id'); + ?> + </fieldset> + <?= $this->Form->button(__('Submit')) ?> + <?= $this->Form->end() ?> + </div> + </div> +</div> diff --git a/templates/Products/index.php b/templates/Products/index.php new file mode 100644 index 0000000..c4a0f52 --- /dev/null +++ b/templates/Products/index.php @@ -0,0 +1,48 @@ +<?php +/** + * @var \App\View\AppView $this + * @var iterable<\Cake\Datasource\EntityInterface> $products + */ +?> +<div class="products index content"> + <?= $this->Html->link(__('New Product'), ['action' => 'add'], ['class' => 'button float-right']) ?> + <h3><?= __('Products') ?></h3> + <div class="table-responsive"> + <table> + <thead> + <tr> + <th><?= $this->Paginator->sort('id') ?></th> + <th><?= $this->Paginator->sort('name') ?></th> + <th><?= $this->Paginator->sort('product_category_id') ?></th> + <th><?= $this->Paginator->sort('product_type_id') ?></th> + <th class="actions"><?= __('Actions') ?></th> + </tr> + </thead> + <tbody> + <?php foreach ($products as $product): ?> + <tr> + <td><?= $this->Number->format($product->id) ?></td> + <td><?= h($product->name) ?></td> + <td><?= $product->hasValue('product_category') ? $this->Html->link($product->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $product->product_category->id]) : '' ?></td> + <td><?= $product->product_type_id->name ?></td> + <td class="actions"> + <?= $this->Html->link(__('View'), ['action' => 'view', $product->id]) ?> + <?= $this->Html->link(__('Edit'), ['action' => 'edit', $product->id]) ?> + <?= $this->Form->postLink(__('Delete'), ['action' => 'delete', $product->id], ['confirm' => __('Are you sure you want to delete # {0}?', $product->id)]) ?> + </td> + </tr> + <?php endforeach; ?> + </tbody> + </table> + </div> + <div class="paginator"> + <ul class="pagination"> + <?= $this->Paginator->first('<< ' . __('first')) ?> + <?= $this->Paginator->prev('< ' . __('previous')) ?> + <?= $this->Paginator->numbers() ?> + <?= $this->Paginator->next(__('next') . ' >') ?> + <?= $this->Paginator->last(__('last') . ' >>') ?> + </ul> + <p><?= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?></p> + </div> +</div> diff --git a/templates/Products/view.php b/templates/Products/view.php new file mode 100644 index 0000000..ee86f4d --- /dev/null +++ b/templates/Products/view.php @@ -0,0 +1,40 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $product + */ +?> +<div class="row"> + <aside class="column"> + <div class="side-nav"> + <h4 class="heading"><?= __('Actions') ?></h4> + <?= $this->Html->link(__('Edit Product'), ['action' => 'edit', $product->id], ['class' => 'side-nav-item']) ?> + <?= $this->Form->postLink(__('Delete Product'), ['action' => 'delete', $product->id], ['confirm' => __('Are you sure you want to delete # {0}?', $product->id), 'class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('List Products'), ['action' => 'index'], ['class' => 'side-nav-item']) ?> + <?= $this->Html->link(__('New Product'), ['action' => 'add'], ['class' => 'side-nav-item']) ?> + </div> + </aside> + <div class="column column-80"> + <div class="products view content"> + <h3><?= h($product->name) ?></h3> + <table> + <tr> + <th><?= __('Name') ?></th> + <td><?= h($product->name) ?></td> + </tr> + <tr> + <th><?= __('Product Category') ?></th> + <td><?= $product->hasValue('product_category') ? $this->Html->link($product->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $product->product_category->id]) : '' ?></td> + </tr> + <tr> + <th><?= __('Id') ?></th> + <td><?= $this->Number->format($product->id) ?></td> + </tr> + <tr> + <th><?= __('Product Type Id') ?></th> + <td><?= $product->product_type_id->name ?></td> + </tr> + </table> + </div> + </div> +</div> diff --git a/templates/element/Layout/submenu.php b/templates/element/Layout/submenu.php new file mode 100644 index 0000000..21480a6 --- /dev/null +++ b/templates/element/Layout/submenu.php @@ -0,0 +1,55 @@ +<?= $this->ActiveLink->link('Catalogs', [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'index', +], [ + 'class' => 'submenu-link', + 'target' => [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + ], +]); ?> +<?= $this->ActiveLink->link('Products', [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'index', +], [ + 'class' => 'submenu-link', + 'target' => [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + ], +]); ?> +<?= $this->ActiveLink->link('Categories', [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'index', +], [ + 'class' => 'submenu-link', + 'target' => [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + ], +]); ?> +<?= $this->ActiveLink->link('Attributes', [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'index', +], [ + 'class' => 'submenu-link', + 'target' => [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + ], +]); ?> +<?= $this->ActiveLink->link('External Catalogs', [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'index', +], [ + 'class' => 'submenu-link', + 'target' => [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + ], +]); ?> diff --git a/templates/element/ProductCategoryAttributes/form.php b/templates/element/ProductCategoryAttributes/form.php new file mode 100644 index 0000000..c6a5c88 --- /dev/null +++ b/templates/element/ProductCategoryAttributes/form.php @@ -0,0 +1,49 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategoryAttribute + * @var \Cake\Collection\CollectionInterface|string[] $productCategories + */ + +$numOptions = !$productCategoryAttribute->hasValue('product_category_attribute_options') || empty($productCategoryAttribute->product_category_attribute_options) ? 0 : count($productCategoryAttribute->product_category_attribute_options); +$cnt = 0; +$prefix = $prefix ?? ''; +?> +<?php + echo $this->Form->control('name'); + echo $this->Form->control('product_category_id', ['options' => $productCategories, 'empty' => true]); + echo $this->Form->control('attribute_type_id'); + echo $this->Form->control('enabled'); +?> +<legend><?= __('Attribute Options') . '<small class="ms-2">' . $this->Html->link('Add Option', '#', [ + 'id' => 'add-option-button', + ]) . '</small>'; ?></legend> +<?= $this->Form->number('prefix', [ + 'value' => $numOptions - 1, + 'id' => 'attribute_options_prefix', + 'style' => 'display:none;', + 'hx-get' => $this->Url->build([ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributeOptions', + 'action' => 'add', + ]), + 'hx-trigger' => 'change', + 'hx-target' => '#attribute-options-container', + 'hx-swap' => 'beforeend', + 'data-test' => 1, +]); ?> + +<div id="attribute-options-container" class="container"> + <?php if ($productCategoryAttribute->hasValue('product_category_attribute_options')) : ?> + <?php + foreach ($productCategoryAttribute->product_category_attribute_options as $attributeOption) { + $prefix = 'product_category_attribute_options.' . $cnt . '.'; + echo '<hr class="my-2">'; + echo $this->element('CakeProducts.ProductCategoryAttributes/product_category_attribute_option_form', [ + 'attributeOption' => $attributeOption, + 'prefix' => $prefix, + ]); + $cnt++; + } ?> + <?php endif; ?> +</div> diff --git a/templates/element/ProductCategoryAttributes/product_category_attribute_option_form.php b/templates/element/ProductCategoryAttributes/product_category_attribute_option_form.php new file mode 100644 index 0000000..ec79239 --- /dev/null +++ b/templates/element/ProductCategoryAttributes/product_category_attribute_option_form.php @@ -0,0 +1,32 @@ +<?php +/** + * @var \App\View\AppView $this + * @var \Cake\Datasource\EntityInterface $productCategoryAttributeOption + */ +$cnt = 0; +$prefix = $prefix ?? ''; +\Cake\Log\Log::debug('$prefix'); +\Cake\Log\Log::debug($prefix); +?> +<div class="container product-category-attribute-options-container" data-test="1" data-prefix="<?= $prefix; ?>"> + <div class="row"> + <div class="col"> + <?= $this->Form->control($prefix . 'attribute_value', [ + 'label' => 'Value', + ]); ?> + </div> + <div class="col"> + <?= $this->Form->control($prefix . 'attribute_label', [ + 'label' => 'Label', + ]); ?> + </div> + <div class="col"> + <?= $this->Form->control($prefix . 'enabled', [ + 'type' => 'checkbox', + 'checked' => true, + 'label' => 'Enabled', + ]); ?> + </div> + </div> + +</div> diff --git a/tests/Fixture/ExternalProductCatalogsFixture.php b/tests/Fixture/ExternalProductCatalogsFixture.php new file mode 100644 index 0000000..8b93827 --- /dev/null +++ b/tests/Fixture/ExternalProductCatalogsFixture.php @@ -0,0 +1,33 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\Fixture; + +use Cake\TestSuite\Fixture\TestFixture; + +/** + * ExternalProductCatalogsFixture + */ +class ExternalProductCatalogsFixture extends TestFixture +{ + /** + * Init method + * + * @return void + */ + public function init(): void + { + $this->records = [ + [ + 'id' => 1, + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'base_url' => 'http://localhost:8766', + 'api_url' => 'http://localhost:8766/api', + 'created' => '2024-11-22 09:39:37', + 'deleted' => '2024-11-22 09:39:37', + 'enabled' => 1, + ], + ]; + parent::init(); + } +} diff --git a/tests/Fixture/ProductCatalogsFixture.php b/tests/Fixture/ProductCatalogsFixture.php new file mode 100644 index 0000000..c189736 --- /dev/null +++ b/tests/Fixture/ProductCatalogsFixture.php @@ -0,0 +1,36 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\Fixture; + +use Cake\TestSuite\Fixture\TestFixture; + +/** + * ProductCatalogsFixture + */ +class ProductCatalogsFixture extends TestFixture +{ + /** + * Init method + * + * @return void + */ + public function init(): void + { + $this->records = [ + [ + 'id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'name' => 'Automotive', + 'catalog_description' => '', + 'enabled' => true, + ], + [ + 'id' => 'f56f3412-ed23-490b-be6e-016208c415d2', + 'name' => 'Software', + 'catalog_description' => '', + 'enabled' => true, + ], + ]; + parent::init(); + } +} diff --git a/tests/Fixture/ProductCategoriesFixture.php b/tests/Fixture/ProductCategoriesFixture.php new file mode 100644 index 0000000..2d3d520 --- /dev/null +++ b/tests/Fixture/ProductCategoriesFixture.php @@ -0,0 +1,101 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\Fixture; + +use Cake\TestSuite\Fixture\TestFixture; + +/** + * ProductCategoriesFixture + */ +class ProductCategoriesFixture extends TestFixture +{ + /** + * Init method + * + * @return void + */ + public function init(): void + { + $this->records = [ + [ + 'id' => 1, + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'internal_id' => 'db4b4273-eddc-46d4-93c8-45cf7c6e058e', + 'name' => 'Engine', + 'category_description' => '', + 'parent_id' => null, + 'lft' => 1, + 'rght' => 4, + 'enabled' => true, + ], + [ + 'id' => 2, + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'internal_id' => '3c2377c5-b97c-4bc9-9660-8f77b4893d8b', + 'name' => 'Engine Internals', + 'category_description' => '', + 'parent_id' => 1, + 'lft' => 2, + 'rght' => 3, + 'enabled' => true, + ], + [ + 'id' => 3, + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'internal_id' => 'fbee6709-396f-4bb4-b60b-e125b0bc4e83', + 'name' => 'Electrical', + 'category_description' => '', + 'parent_id' => null, + 'lft' => 5, + 'rght' => 8, + 'enabled' => true, + ], + [ + 'id' => 4, + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'internal_id' => '6d223283-361b-4f9f-a7f1-c97aa0ca4c23', + 'name' => 'Wiring', + 'category_description' => '', + 'parent_id' => 3, + 'lft' => 6, + 'rght' => 7, + 'enabled' => true, + ], + [ + 'id' => 5, + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'internal_id' => 'c447b6f4-0fb1-4d59-ba45-5613829a725a', + 'name' => 'Suspension', + 'category_description' => '', + 'parent_id' => null, + 'lft' => 9, + 'rght' => 12, + 'enabled' => true, + ], + [ + 'id' => 6, + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'internal_id' => '1e749d3b-aee0-48a5-8d6c-8cf2b83e9b6e', + 'name' => 'Coilovers', + 'category_description' => '', + 'parent_id' => 5, + 'lft' => 10, + 'rght' => 11, + 'enabled' => true, + ], + [ + 'id' => 7, + 'product_catalog_id' => 'f56f3412-ed23-490b-be6e-016208c415d2', + 'internal_id' => '8c89a3ca-d56f-46bf-a738-7e85b3342b2a', + 'name' => 'Support', + 'category_description' => '', + 'parent_id' => null, + 'lft' => 1, + 'rght' => 2, + 'enabled' => true, + ], + ]; + parent::init(); + } +} diff --git a/tests/Fixture/ProductCategoryAttributeOptionsFixture.php b/tests/Fixture/ProductCategoryAttributeOptionsFixture.php new file mode 100644 index 0000000..0a53303 --- /dev/null +++ b/tests/Fixture/ProductCategoryAttributeOptionsFixture.php @@ -0,0 +1,31 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\Fixture; + +use Cake\TestSuite\Fixture\TestFixture; + +/** + * ProductCategoryAttributeOptionsFixture + */ +class ProductCategoryAttributeOptionsFixture extends TestFixture +{ + /** + * Init method + * + * @return void + */ + public function init(): void + { + $this->records = [ + [ + 'id' => 'e06f1723-2456-483a-b3c4-004603e032a8', + 'product_category_attribute_id' => '37078cf0-0130-4b93-bb7e-abe7d665ed2c', + 'attribute_value' => 'Lorem ipsum dolor sit amet', + 'attribute_label' => 'Lorem ipsum dolor sit amet', + 'enabled' => 1, + ], + ]; + parent::init(); + } +} diff --git a/tests/Fixture/ProductCategoryAttributesFixture.php b/tests/Fixture/ProductCategoryAttributesFixture.php new file mode 100644 index 0000000..1f77437 --- /dev/null +++ b/tests/Fixture/ProductCategoryAttributesFixture.php @@ -0,0 +1,31 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\Fixture; + +use Cake\TestSuite\Fixture\TestFixture; + +/** + * ProductCategoryAttributesFixture + */ +class ProductCategoryAttributesFixture extends TestFixture +{ + /** + * Init method + * + * @return void + */ + public function init(): void + { + $this->records = [ + [ + 'id' => '37078cf0-0130-4b93-bb7e-abe7d665ed2c', + 'name' => 'Color', + 'product_category_id' => '6d223283-361b-4f9f-a7f1-c97aa0ca4c23', + 'attribute_type_id' => 1, + 'enabled' => 1, + ], + ]; + parent::init(); + } +} diff --git a/tests/Fixture/ProductsFixture.php b/tests/Fixture/ProductsFixture.php new file mode 100644 index 0000000..5f55d74 --- /dev/null +++ b/tests/Fixture/ProductsFixture.php @@ -0,0 +1,30 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\Fixture; + +use Cake\TestSuite\Fixture\TestFixture; + +/** + * ProductsFixture + */ +class ProductsFixture extends TestFixture +{ + /** + * Init method + * + * @return void + */ + public function init(): void + { + $this->records = [ + [ + 'id' => 'cfc98a9a-29b2-44c8-b587-8156adc05317', + 'name' => '12AWG RED TXL Wire', + 'product_category_id' => '6d223283-361b-4f9f-a7f1-c97aa0ca4c23', + 'product_type_id' => 1, + ], + ]; + parent::init(); + } +} diff --git a/tests/TestCase/Controller/BaseControllerTest.php b/tests/TestCase/Controller/BaseControllerTest.php new file mode 100644 index 0000000..eae19fc --- /dev/null +++ b/tests/TestCase/Controller/BaseControllerTest.php @@ -0,0 +1,25 @@ +<?php + +namespace CakeProducts\Test\TestCase\Controller; + +use Cake\TestSuite\IntegrationTestTrait; +use Cake\TestSuite\TestCase; + +class BaseControllerTest extends TestCase +{ + use IntegrationTestTrait; + + public function loginUserByRole(string $role = 'admin'): void + { + $this->session(['Auth.User.id' => 1]); + $this->session(['Auth.id' => 1]); + } + + /** + * @return void + */ + public function testTest() + { + $this->assertEquals(1, 1); + } +} diff --git a/tests/TestCase/Controller/ExternalProductCatalogsControllerTest.php b/tests/TestCase/Controller/ExternalProductCatalogsControllerTest.php new file mode 100644 index 0000000..02ab223 --- /dev/null +++ b/tests/TestCase/Controller/ExternalProductCatalogsControllerTest.php @@ -0,0 +1,446 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Controller; + +use Cake\ORM\Table; +use Cake\TestSuite\IntegrationTestTrait; +use Cake\TestSuite\TestCase; +use CakeProducts\Controller\ExternalProductCatalogsController; +use CakeProducts\Model\Table\ExternalProductCatalogsTable; +use PHPUnit\Exception; + +/** + * CakeProducts\Controller\ExternalProductCatalogsController Test Case + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController + */ +class ExternalProductCatalogsControllerTest extends BaseControllerTest +{ + /** + * Test subject table + * + * @var ExternalProductCatalogsTable|Table + */ + protected $ExternalProductCatalogs; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ExternalProductCatalogs', + 'plugin.CakeProducts.ProductCatalogs', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $this->enableCsrfToken(); + $this->enableSecurityToken(); + $this->ExternalProductCatalogs = $this->getTableLocator()->get('ExternalProductCatalogs'); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ExternalProductCatalogs); + + parent::tearDown(); + } + + /** + * Test index method + * + * Tests the index action with an unauthenticated user (not logged in) + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::index() + * @throws Exception + * + * @return void + */ + public function testIndexGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test index method + * + * Tests the index action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::index() + * @throws Exception + * + * @return void + */ + public function testIndexGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test view method + * + * Tests the view action with an unauthenticated user (not logged in) + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::view() + * @throws Exception + * + * @return void + */ + public function testViewGetUnauthenticated(): void + { + $id = 1; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test view method + * + * Tests the view action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::view() + * @throws Exception + * + * @return void + */ + public function testViewGetLoggedIn(): void + { + $id = 1; + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test add method + * + * Tests the add action with an unauthenticated user (not logged in) + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddGetUnauthenticated(): void + { + $cntBefore = $this->ExternalProductCatalogs->find()->count(); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ExternalProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests the add action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddGetLoggedIn(): void + { + $cntBefore = $this->ExternalProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(200); + + $cntAfter = $this->ExternalProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddPostLoggedInSuccess(): void + { + $cntBefore = $this->ExternalProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'add', + ]; + $data = [ + 'product_catalog_id' => 'f56f3412-ed23-490b-be6e-016208c415d2', + 'base_url' => 'http://localhost:8766', + 'api_url' => 'http://localhost:8766/api/v1/', + 'enabled' => true, + ]; + $this->post($url, $data); + $this->assertResponseCode(302); + $this->assertRedirectContains('external-product-catalogs'); + + $cntAfter = $this->ExternalProductCatalogs->find()->count(); + $this->assertEquals($cntBefore + 1, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddPostLoggedInFailure(): void + { + $cntBefore = $this->ExternalProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'add', + ]; + $data = [ + 'product_catalog_id' => 999999, + 'base_url' => '', + 'api_url' => 'http://localhost:8766/api/v1/', + 'enabled' => true, + ]; + $this->post($url, $data); + $this->assertResponseCode(200); + + $cntAfter = $this->ExternalProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test edit method + * + * Tests the edit action with an unauthenticated user (not logged in) + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'edit', + 1, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test edit method + * + * Tests the edit action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'edit', + 1, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditPutLoggedInSuccess(): void + { + $this->loginUserByRole('admin'); + $id = 1; + $before = $this->ExternalProductCatalogs->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'edit', + $id, + ]; + $data = [ + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'base_url' => 'http://localhost:8766', + 'api_url' => 'http://localhost:8766/api/v1/', + 'enabled' => true, + ]; + $this->put($url, $data); + + $this->assertResponseCode(302); + $this->assertRedirectContains('external-product-catalogs'); + + $after = $this->ExternalProductCatalogs->get($id); + // assert saved properly below + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditPutLoggedInFailure(): void + { + $this->loginUserByRole('admin'); + $id = 1; + $before = $this->ExternalProductCatalogs->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'edit', + $id, + ]; + $data = [ + 'product_catalog_id' => 9999999, + 'base_url' => '', + 'api_url' => 'http://localhost:8766/api/v1/', + 'enabled' => true, + ]; + $this->put($url, $data); + $this->assertResponseCode(200); + $after = $this->ExternalProductCatalogs->get($id); + + // assert save failed below + } + + /** + * Test delete method + * + * Tests the delete action with an unauthenticated user (not logged in) + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::delete() + * @throws Exception + * + * @return void + */ + public function testDeleteUnauthenticated(): void + { + $cntBefore = $this->ExternalProductCatalogs->find()->count(); + + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'delete', + 1, + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ExternalProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test delete method + * + * Tests the delete action with a logged in user + * + * @uses \CakeProducts\Controller\ExternalProductCatalogsController::delete() + * @throws Exception + * + * @return void + */ + public function testDeleteLoggedIn(): void + { + $cntBefore = $this->ExternalProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ExternalProductCatalogs', + 'action' => 'delete', + 1, + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('external-product-catalogs'); + + $cntAfter = $this->ExternalProductCatalogs->find()->count(); + $this->assertEquals($cntBefore - 1, $cntAfter); + } +} diff --git a/tests/TestCase/Controller/ProductCatalogsControllerTest.php b/tests/TestCase/Controller/ProductCatalogsControllerTest.php new file mode 100644 index 0000000..ffe2423 --- /dev/null +++ b/tests/TestCase/Controller/ProductCatalogsControllerTest.php @@ -0,0 +1,450 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Controller; + +use Cake\ORM\Table; +use Cake\TestSuite\IntegrationTestTrait; +use Cake\TestSuite\TestCase; +use CakeProducts\Controller\ProductCatalogsController; +use CakeProducts\Model\Table\ProductCatalogsTable; +use PHPUnit\Exception; + +/** + * CakeProducts\Controller\ProductCatalogsController Test Case + * + * @uses \CakeProducts\Controller\ProductCatalogsController + */ +class ProductCatalogsControllerTest extends BaseControllerTest +{ + /** + * Test subject table + * + * @var ProductCatalogsTable|Table + */ + protected $ProductCatalogs; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCatalogs', + 'plugin.CakeProducts.ProductCategories', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $this->enableCsrfToken(); + $this->enableSecurityToken(); + $config = $this->getTableLocator()->exists('ProductCatalogs') ? [] : ['className' => ProductCatalogsTable::class]; + $this->ProductCatalogs = $this->getTableLocator()->get('ProductCatalogs', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCatalogs); + + parent::tearDown(); + } + + /** + * Test index method + * + * Tests the index action with an unauthenticated user (not logged in) + * + * @uses ProductCatalogsController::index() + * @throws Exception + * + * @return void + */ + public function testIndexGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test index method + * + * Tests the index action with a logged in user + * + * @uses \CakeProducts\Controller\ProductCatalogsController::index() + * @throws Exception + * + * @return void + */ + public function testIndexGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test view method + * + * Tests the view action with an unauthenticated user (not logged in) + * + * @uses ProductCatalogsController::view() + * @throws Exception + * + * @return void + */ + public function testViewGetUnauthenticated(): void + { + $id = '115153f3-2f59-4234-8ff8-e1b205761428'; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test view method + * + * Tests the view action with a logged in user + * + * @uses ProductCatalogsController::view() + * @throws Exception + * + * @return void + */ + public function testViewGetLoggedIn(): void + { + $id = '115153f3-2f59-4234-8ff8-e1b205761428'; + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test add method + * + * Tests the add action with an unauthenticated user (not logged in) + * + * @uses ProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddGetUnauthenticated(): void + { + $cntBefore = $this->ProductCatalogs->find()->count(); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests the add action with a logged in user + * + * @uses ProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddGetLoggedIn(): void + { + $cntBefore = $this->ProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @uses ProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddPostLoggedInSuccess(): void + { + $cntBefore = $this->ProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'add', + ]; + $data = [ + 'name' => 'new catalog', + 'catalog_description' => 'description', + 'enabled' => true, + ]; + $this->post($url, $data); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-catalogs'); + + $cntAfter = $this->ProductCatalogs->find()->count(); + $this->assertEquals($cntBefore + 1, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @uses ProductCatalogsController::add() + * @throws Exception + * + * @return void + */ + public function testAddPostLoggedInFailure(): void + { + $cntBefore = $this->ProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'add', + ]; + $data = [ + 'name' => '', + 'catalog_description' => '', + 'enabled' => '', + ]; + $this->post($url, $data); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test edit method + * + * Tests the edit action with an unauthenticated user (not logged in) + * + * @uses ProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditGetUnauthenticated(): void + { + $id = '115153f3-2f59-4234-8ff8-e1b205761428'; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'edit', + $id, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test edit method + * + * Tests the edit action with a logged in user + * + * @uses ProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $id = '115153f3-2f59-4234-8ff8-e1b205761428'; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'edit', + $id, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @uses ProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditPutLoggedInSuccess(): void + { + $this->loginUserByRole('admin'); + $id = '115153f3-2f59-4234-8ff8-e1b205761428'; +// $before = $this->ProductCatalogs->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'edit', + $id, + ]; + $data = [ + // test new data here + 'name' => 'edited name', + 'catalog_description' => 'new catalog description', + 'enabled' => true, + ]; + $this->put($url, $data); + + $this->assertResponseCode(302); + $this->assertRedirectContains('product-catalogs'); + + $after = $this->ProductCatalogs->get($id); + $this->assertEquals($data['name'], $after->name); + $this->assertEquals($data['catalog_description'], $after->catalog_description); + // assert saved properly below + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @uses ProductCatalogsController::edit() + * @throws Exception + * + * @return void + */ + public function testEditPutLoggedInFailure(): void + { + $this->loginUserByRole('admin'); + $id = '115153f3-2f59-4234-8ff8-e1b205761428'; + $before = $this->ProductCatalogs->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'edit', + $id, + ]; + $data = [ + 'name' => '', + 'catalog_description' => 'edited description', + 'enabled' => '', + ]; + $this->put($url, $data); + $this->assertResponseCode(200); + $after = $this->ProductCatalogs->get($id); + $this->assertEquals($before->name, $after->name); + $this->assertEquals($before->catalog_description, $after->catalog_description); + // assert save failed below + } + + /** + * Test delete method + * + * Tests the delete action with an unauthenticated user (not logged in) + * + * @uses ProductCatalogsController::delete() + * @throws Exception + * + * @return void + */ + public function testDeleteUnauthenticated(): void + { + $cntBefore = $this->ProductCatalogs->find()->count(); + + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'delete', + 1, + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCatalogs->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test delete method + * + * Tests the delete action with a logged in user + * + * @uses ProductCatalogsController::delete() + * @throws Exception + * + * @return void + */ + public function testDeleteLoggedIn(): void + { + $cntBefore = $this->ProductCatalogs->find()->count(); + + $this->loginUserByRole('admin'); + $id = '115153f3-2f59-4234-8ff8-e1b205761428'; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCatalogs', + 'action' => 'delete', + $id, + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-catalogs'); + + $cntAfter = $this->ProductCatalogs->find()->count(); + $this->assertEquals($cntBefore - 1, $cntAfter); + } +} diff --git a/tests/TestCase/Controller/ProductCategoriesControllerTest.php b/tests/TestCase/Controller/ProductCategoriesControllerTest.php new file mode 100644 index 0000000..1808f4e --- /dev/null +++ b/tests/TestCase/Controller/ProductCategoriesControllerTest.php @@ -0,0 +1,455 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Controller; + +use Cake\ORM\Table; +use Cake\TestSuite\IntegrationTestTrait; +use Cake\TestSuite\TestCase; +use CakeProducts\Controller\ProductCategoriesController; +use CakeProducts\Model\Table\ProductCategoriesTable; +use PHPUnit\Exception; + +/** + * CakeProducts\Controller\ProductCategoriesController Test Case + * + * @uses ProductCategoriesController + */ +class ProductCategoriesControllerTest extends BaseControllerTest +{ + /** + * Test subject table + * + * @var ProductCategoriesTable|Table + */ + protected $ProductCategories; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCatalogs', + 'plugin.CakeProducts.ProductCategories', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $this->enableCsrfToken(); + $this->enableSecurityToken(); + $this->ProductCategories = $this->getTableLocator()->get('ProductCategories'); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCategories); + + parent::tearDown(); + } + + /** + * Test index method + * + * Tests the index action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::index + */ + public function testIndexGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test index method + * + * Tests the index action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::index + */ + public function testIndexGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test view method + * + * Tests the view action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::view + */ + public function testViewGetUnauthenticated(): void + { + $id = 1; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test view method + * + * Tests the view action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::view + */ + public function testViewGetLoggedIn(): void + { + $id = 1; + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test add method + * + * Tests the add action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::add + */ + public function testAddGetUnauthenticated(): void + { + $cntBefore = $this->ProductCategories->find()->count(); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCategories->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::add + */ + public function testAddGetLoggedIn(): void + { + $cntBefore = $this->ProductCategories->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCategories->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::add + */ + public function testAddPostLoggedInSuccess(): void + { + $cntBefore = $this->ProductCategories->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'add', + ]; + $data = [ + 'name' => 'Electrical Plugs', + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'category_description' => 'electrical', + 'parent_id' => 3, + 'enabled' => true, + ]; + $this->post($url, $data); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-categories'); + + $cntAfter = $this->ProductCategories->find()->count(); + $this->assertEquals($cntBefore + 1, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::add + */ + public function testAddPostLoggedInFailure(): void + { + $cntBefore = $this->ProductCategories->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'add', + ]; + $data = [ + 'name' => '', + 'product_catalog_id' => '', + 'category_description' => 'electrical', + 'parent_id' => '', + 'enabled' => true, + ]; + $this->post($url, $data); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCategories->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test edit method + * + * Tests the edit action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::edit + */ + public function testEditGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'edit', + 1, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test edit method + * + * Tests the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::edit + */ + public function testEditGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'edit', + 1, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::edit + */ + public function testEditPutLoggedInSuccess(): void + { + $this->loginUserByRole('admin'); + $id = 1; + $before = $this->ProductCategories->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'edit', + $id, + ]; + $data = [ + // test new data here + 'name' => 'Electrical v2', + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'category_description' => 'electrical v2', + 'parent_id' => '', + 'enabled' => true, + ]; + $this->put($url, $data); + + $this->assertResponseCode(302); + $this->assertRedirectContains('product-categories'); + + $after = $this->ProductCategories->get($id); + $this->assertEquals($data['name'], $after->name); + $this->assertEquals($data['product_catalog_id'], $after->product_catalog_id); + $this->assertEquals($data['category_description'], $after->category_description); + $this->assertNull($after->parent_id); + // assert saved properly below + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::edit + */ + public function testEditPutLoggedInFailure(): void + { + $this->loginUserByRole('admin'); + $id = 1; + $before = $this->ProductCategories->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'edit', + $id, + ]; + $data = [ + 'name' => '', + 'product_catalog_id' => '', + 'category_description' => 'electrical', + 'parent_id' => '', + 'enabled' => true, + ]; + $this->put($url, $data); + $this->assertResponseCode(200); + $after = $this->ProductCategories->get($id); + + // assert save failed below + } + + /** + * Test delete method + * + * Tests the delete action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoriesController::delete + */ + public function testDeleteUnauthenticated(): void + { + $cntBefore = $this->ProductCategories->find()->count(); + + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'delete', + 1, + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCategories->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test delete method + * + * Tests the delete action with a logged in user + * + * @return void + *@throws Exception + * + * @uses ProductCategoriesController::delete + */ + public function testDeleteLoggedIn(): void + { + $cntBefore = $this->ProductCategories->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategories', + 'action' => 'delete', + 1, + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-categories'); + + $cntAfter = $this->ProductCategories->find()->count(); + $this->assertEquals($cntBefore - 1, $cntAfter); + } +} diff --git a/tests/TestCase/Controller/ProductCategoryAttributeOptionsControllerTest.php b/tests/TestCase/Controller/ProductCategoryAttributeOptionsControllerTest.php new file mode 100644 index 0000000..05fb7b4 --- /dev/null +++ b/tests/TestCase/Controller/ProductCategoryAttributeOptionsControllerTest.php @@ -0,0 +1,198 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Controller; + +use Cake\ORM\Table; +use CakeProducts\Controller\ProductCategoryAttributeOptionsController; +use CakeProducts\Model\Table\ProductCategoryAttributeOptionsTable; +use PHPUnit\Exception; + +/** + * CakeProducts\Controller\ProductCategoryAttributeOptionsController Test Case + * + * @uses \CakeProducts\Controller\ProductCategoryAttributeOptionsController + */ +class ProductCategoryAttributeOptionsControllerTest extends BaseControllerTest +{ + /** + * Test subject + * + * @var ProductCategoryAttributeOptionsTable|Table + */ + protected $ProductCategoryAttributeOptions; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCategoryAttributeOptions', + 'plugin.CakeProducts.ProductCategoryAttributes', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $this->enableCsrfToken(); + $this->enableSecurityToken(); + $config = $this->getTableLocator()->exists('ProductCategoryAttributeOptions') ? [] : ['className' => ProductCategoryAttributeOptionsTable::class]; + $this->ProductCategoryAttributeOptions = $this->getTableLocator()->get('ProductCategoryAttributeOptions', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCategoryAttributeOptions); + + parent::tearDown(); + } + + /** + * Test add method + * + * Tests the add action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses CustomersContactsController::add + */ + public function testAddGetUnauthenticated(): void + { + $cntBefore = $this->ProductCategoryAttributeOptions->find()->count(); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributeOptions', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCategoryAttributeOptions->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses CustomersContactsController::add + */ + public function testAddGetLoggedIn(): void + { + $cntBefore = $this->ProductCategoryAttributeOptions->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributeOptions', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCategoryAttributeOptions->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses CustomersContactsController::add + */ + public function testAddPostLoggedInHasNoEffect(): void + { + $cntBefore = $this->ProductCategoryAttributeOptions->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributeOptions', + 'action' => 'add', + ]; + $data = []; + $this->post($url, $data); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCategoryAttributeOptions->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test delete method + * + * Tests the delete action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses CustomersContactsController::delete + */ + public function testDeleteUnauthenticated(): void + { + $cntBefore = $this->ProductCategoryAttributeOptions->find()->count(); + + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributeOptions', + 'action' => 'delete', + 'e06f1723-2456-483a-b3c4-004603e032a8', + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCategoryAttributeOptions->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test delete method + * + * Tests the delete action with a logged in user + * + * @return void + *@throws Exception + * + * @uses CustomersContactsController::delete + */ + public function testDeleteLoggedIn(): void + { + $cntBefore = $this->ProductCategoryAttributeOptions->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributeOptions', + 'action' => 'delete', + 'e06f1723-2456-483a-b3c4-004603e032a8', + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-category-attributes'); + + $cntAfter = $this->ProductCategoryAttributeOptions->find()->count(); + $this->assertEquals($cntBefore - 1, $cntAfter); + } +} diff --git a/tests/TestCase/Controller/ProductCategoryAttributesControllerTest.php b/tests/TestCase/Controller/ProductCategoryAttributesControllerTest.php new file mode 100644 index 0000000..f79fdf4 --- /dev/null +++ b/tests/TestCase/Controller/ProductCategoryAttributesControllerTest.php @@ -0,0 +1,498 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Controller; + +use Cake\ORM\Table; +use Cake\TestSuite\IntegrationTestTrait; +use Cake\TestSuite\TestCase; +use CakeProducts\Controller\ProductCategoryAttributesController; +use CakeProducts\Model\Table\ProductCategoryAttributesTable; +use PHPUnit\Exception; + +/** + * CakeProducts\Controller\ProductCategoryAttributesController Test Case + * + * @uses ProductCategoryAttributesController + */ +class ProductCategoryAttributesControllerTest extends BaseControllerTest +{ + /** + * Test subject + * + * @var ProductCategoryAttributesTable|Table + */ + protected $ProductCategoryAttributes; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCategoryAttributes', + 'plugin.CakeProducts.ProductCategories', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $this->enableCsrfToken(); + $this->enableSecurityToken(); + $config = $this->getTableLocator()->exists('ProductCategoryAttributes') ? [] : ['className' => ProductCategoryAttributesTable::class]; + $this->ProductCategoryAttributes = $this->getTableLocator()->get('ProductCategoryAttributes', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCategoryAttributes); + + parent::tearDown(); + } + + /** + * Test index method + * + * Tests the index action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::index + */ + public function testIndexGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test index method + * + * Tests the index action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::index + */ + public function testIndexGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test view method + * + * Tests the view action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::view + */ + public function testViewGetUnauthenticated(): void + { + $id = '37078cf0-0130-4b93-bb7e-abe7d665ed2c'; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test view method + * + * Tests the view action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::view + */ + public function testViewGetLoggedIn(): void + { + $id = '37078cf0-0130-4b93-bb7e-abe7d665ed2c'; + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test add method + * + * Tests the add action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::add + */ + public function testAddGetUnauthenticated(): void + { + $cntBefore = $this->ProductCategoryAttributes->find()->count(); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCategoryAttributes->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::add + */ + public function testAddGetLoggedIn(): void + { + $cntBefore = $this->ProductCategoryAttributes->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCategoryAttributes->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::add + */ + public function testAddPostLoggedInSuccess(): void + { + $cntBefore = $this->ProductCategoryAttributes->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'add', + ]; + $data = [ + 'name' => 'Size', + 'product_category_id' => 'db4b4273-eddc-46d4-93c8-45cf7c6e058e', + 'attribute_type_id' => 2, + 'enabled' => true, + ]; + $this->post($url, $data); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-category-attributes'); + + $cntAfter = $this->ProductCategoryAttributes->find()->count(); + $this->assertEquals($cntBefore + 1, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::add + */ + public function testAddPostLoggedInSuccessConstrainedWithOptions(): void + { + $cntBefore = $this->ProductCategoryAttributes->find()->count(); + $cntOptionsBefore = $this->ProductCategoryAttributes->ProductCategoryAttributeOptions->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'add', + ]; + $data = [ + 'name' => 'Size', + 'product_category_id' => 'db4b4273-eddc-46d4-93c8-45cf7c6e058e', + 'attribute_type_id' => 1, + 'enabled' => true, + 'product_category_attribute_options' => [ + [ + 'attribute_value' => 'XL', + 'attribute_label' => 'XL', + 'enabled' => true, + ], + [ + 'attribute_value' => 'L', + 'attribute_label' => 'L', + 'enabled' => true, + ] + ], + ]; + $this->post($url, $data); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-category-attributes'); + + $cntAfter = $this->ProductCategoryAttributes->find()->count(); + $cntOptionsAfter = $this->ProductCategoryAttributes->ProductCategoryAttributeOptions->find()->count(); + + $this->assertEquals($cntBefore + 1, $cntAfter); + $this->assertEquals($cntOptionsBefore + 2, $cntOptionsAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::add + */ + public function testAddPostLoggedInFailure(): void + { + $cntBefore = $this->ProductCategoryAttributes->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'add', + ]; + $data = [ + 'name' => '', + 'product_category_id' => 1, + 'attribute_type_id' => 1, + 'enabled' => true, + ]; + $this->post($url, $data); + $this->assertResponseCode(200); + + $cntAfter = $this->ProductCategoryAttributes->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test edit method + * + * Tests the edit action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::edit + */ + public function testEditGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'edit', + '37078cf0-0130-4b93-bb7e-abe7d665ed2c', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test edit method + * + * Tests the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::edit + */ + public function testEditGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'edit', + '37078cf0-0130-4b93-bb7e-abe7d665ed2c', + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::edit + */ + public function testEditPutLoggedInSuccess(): void + { + $this->loginUserByRole('admin'); + $id = '37078cf0-0130-4b93-bb7e-abe7d665ed2c'; + $before = $this->ProductCategoryAttributes->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'edit', + $id, + ]; + $data = [ + // test new data here + 'name' => 'Color', + 'product_category_id' => 'db4b4273-eddc-46d4-93c8-45cf7c6e058e', + 'attribute_type_id' => 1, + 'enabled' => true, + ]; + $this->put($url, $data); + + $this->assertResponseCode(302); + $this->assertRedirectContains('product-category-attributes'); + + $after = $this->ProductCategoryAttributes->get($id); + // assert saved properly below + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::edit + */ + public function testEditPutLoggedInFailure(): void + { + $this->loginUserByRole('admin'); + $id = '37078cf0-0130-4b93-bb7e-abe7d665ed2c'; + $before = $this->ProductCategoryAttributes->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'edit', + $id, + ]; + $data = [ + 'name' => '', + 'product_category_id' => 1, + 'attribute_type_id' => 1, + 'enabled' => true, + ]; + $this->put($url, $data); + $this->assertResponseCode(200); + $after = $this->ProductCategoryAttributes->get($id); + + // assert save failed below + } + + /** + * Test delete method + * + * Tests the delete action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductCategoryAttributesController::delete + */ + public function testDeleteUnauthenticated(): void + { + $cntBefore = $this->ProductCategoryAttributes->find()->count(); + + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'delete', + '37078cf0-0130-4b93-bb7e-abe7d665ed2c', + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->ProductCategoryAttributes->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test delete method + * + * Tests the delete action with a logged in user + * + * @return void + *@throws Exception + * + * @uses ProductCategoryAttributesController::delete + */ + public function testDeleteLoggedIn(): void + { + $cntBefore = $this->ProductCategoryAttributes->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'ProductCategoryAttributes', + 'action' => 'delete', + '37078cf0-0130-4b93-bb7e-abe7d665ed2c', + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('product-category-attributes'); + + $cntAfter = $this->ProductCategoryAttributes->find()->count(); + $this->assertEquals($cntBefore - 1, $cntAfter); + } +} diff --git a/tests/TestCase/Controller/ProductsControllerTest.php b/tests/TestCase/Controller/ProductsControllerTest.php new file mode 100644 index 0000000..0517918 --- /dev/null +++ b/tests/TestCase/Controller/ProductsControllerTest.php @@ -0,0 +1,452 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Controller; + +use Cake\ORM\Table; +use Cake\TestSuite\IntegrationTestTrait; +use Cake\TestSuite\TestCase; +use CakeProducts\Controller\ProductsController; +use CakeProducts\Model\Table\ProductCatalogsTable; +use CakeProducts\Model\Table\ProductsTable; +use PHPUnit\Exception; + +/** + * CakeProducts\Controller\ProductsController Test Case + * + * @uses ProductsController + */ +class ProductsControllerTest extends BaseControllerTest +{ + /** + * Test subject table + * + * @var ProductsTable|Table + */ + protected $Products; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.Products', + 'plugin.CakeProducts.ProductCategories', +// 'plugin.CakeProducts.ProductCatalogs', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $this->enableCsrfToken(); + $this->enableSecurityToken(); + $config = $this->getTableLocator()->exists('Products') ? [] : ['className' => ProductsTable::class]; + $this->Products = $this->getTableLocator()->get('Products', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->Products); + + parent::tearDown(); + } + + /** + * Test index method + * + * Tests the index action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductsController::index + */ + public function testIndexGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test index method + * + * Tests the index action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::index + */ + public function testIndexGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'index', + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test view method + * + * Tests the view action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductsController::view + */ + public function testViewGetUnauthenticated(): void + { + $id = 'cfc98a9a-29b2-44c8-b587-8156adc05317'; + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test view method + * + * Tests the view action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::view + */ + public function testViewGetLoggedIn(): void + { + $id = 'cfc98a9a-29b2-44c8-b587-8156adc05317'; + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'view', + $id, + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test add method + * + * Tests the add action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductsController::add + */ + public function testAddGetUnauthenticated(): void + { + $cntBefore = $this->Products->find()->count(); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->Products->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::add + */ + public function testAddGetLoggedIn(): void + { + $cntBefore = $this->Products->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'add', + ]; + $this->get($url); + $this->assertResponseCode(200); + + $cntAfter = $this->Products->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::add + */ + public function testAddPostLoggedInSuccess(): void + { + $cntBefore = $this->Products->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'add', + ]; + $data = [ + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'product_category_id' => '6d223283-361b-4f9f-a7f1-c97aa0ca4c23', + 'name' => '16AWG WIRE RED', + 'product_type_id' => 1, + ]; + $this->post($url, $data); + $this->assertResponseCode(302); + $this->assertRedirectContains('products'); + + $cntAfter = $this->Products->find()->count(); + $this->assertEquals($cntBefore + 1, $cntAfter); + } + + /** + * Test add method + * + * Tests a POST request to the add action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::add + */ + public function testAddPostLoggedInFailure(): void + { + $cntBefore = $this->Products->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'add', + ]; + $data = [ + 'product_catalog_id' => '', + 'product_category_id' => '', + 'name' => '', + 'product_type_id' => 1, + ]; + $this->post($url, $data); + $this->assertResponseCode(200); + + $cntAfter = $this->Products->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test edit method + * + * Tests the edit action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductsController::edit + */ + public function testEditGetUnauthenticated(): void + { + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'edit', + 'cfc98a9a-29b2-44c8-b587-8156adc05317', + ]; + $this->get($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + } + + /** + * Test edit method + * + * Tests the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::edit + */ + public function testEditGetLoggedIn(): void + { + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'edit', + 'cfc98a9a-29b2-44c8-b587-8156adc05317', + ]; + $this->get($url); + $this->assertResponseCode(200); + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::edit + */ + public function testEditPutLoggedInSuccess(): void + { + $this->loginUserByRole('admin'); + $id = 'cfc98a9a-29b2-44c8-b587-8156adc05317'; + $before = $this->Products->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'edit', + $id, + ]; + $data = [ + // test new data here + 'product_catalog_id' => '115153f3-2f59-4234-8ff8-e1b205761428', + 'product_category_id' => '6d223283-361b-4f9f-a7f1-c97aa0ca4c23', + 'name' => 'edited product name', + 'product_type_id' => 1, + ]; + $this->put($url, $data); + + $this->assertResponseCode(302); + $this->assertRedirectContains('products'); + + $after = $this->Products->get($id); + $this->assertEquals($data['name'], $after->name); + // assert saved properly below + } + + /** + * Test edit method + * + * Tests a PUT request to the edit action with a logged in user + * + * @return void + * @throws Exception + * + * @uses ProductsController::edit + */ + public function testEditPutLoggedInFailure(): void + { + $this->loginUserByRole('admin'); + $id = 'cfc98a9a-29b2-44c8-b587-8156adc05317'; + $before = $this->Products->get($id); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'edit', + $id, + ]; + $data = [ + 'product_catalog_id' => '', + 'product_category_id' => '', + 'name' => 'edited name not gonna take', + 'product_type_id' => 1, + ]; + $this->put($url, $data); + $this->assertResponseCode(200); + $after = $this->Products->get($id); + $this->assertEquals($before->name, $after->name); + $this->assertEquals($before->product_category_id, $after->product_category_id); + // assert save failed below + } + + /** + * Test delete method + * + * Tests the delete action with an unauthenticated user (not logged in) + * + * @return void + * @throws Exception + * + * @uses ProductsController::delete + */ + public function testDeleteUnauthenticated(): void + { + $cntBefore = $this->Products->find()->count(); + + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'delete', + 'cfc98a9a-29b2-44c8-b587-8156adc05317', + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('login'); + + $cntAfter = $this->Products->find()->count(); + $this->assertEquals($cntBefore, $cntAfter); + } + + /** + * Test delete method + * + * Tests the delete action with a logged in user + * + * @return void + *@throws Exception + * + * @uses ProductsController::delete + */ + public function testDeleteLoggedIn(): void + { + $cntBefore = $this->Products->find()->count(); + + $this->loginUserByRole('admin'); + $url = [ + 'plugin' => 'CakeProducts', + 'controller' => 'Products', + 'action' => 'delete', + 'cfc98a9a-29b2-44c8-b587-8156adc05317', + ]; + $this->delete($url); + $this->assertResponseCode(302); + $this->assertRedirectContains('products'); + + $cntAfter = $this->Products->find()->count(); + $this->assertEquals($cntBefore - 1, $cntAfter); + } +} diff --git a/tests/TestCase/Model/Table/ExternalProductCatalogsTableTest.php b/tests/TestCase/Model/Table/ExternalProductCatalogsTableTest.php new file mode 100644 index 0000000..168ae37 --- /dev/null +++ b/tests/TestCase/Model/Table/ExternalProductCatalogsTableTest.php @@ -0,0 +1,107 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Model\Table; + +use Cake\TestSuite\TestCase; +use CakeProducts\Model\Table\ExternalProductCatalogsTable; + +/** + * CakeProducts\Model\Table\ExternalProductCatalogsTable Test Case + */ +class ExternalProductCatalogsTableTest extends TestCase +{ + /** + * Test subject + * + * @var \CakeProducts\Model\Table\ExternalProductCatalogsTable + */ + protected $ExternalProductCatalogs; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ExternalProductCatalogs', + 'plugin.CakeProducts.ProductCatalogs', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $config = $this->getTableLocator()->exists('ExternalProductCatalogs') ? [] : ['className' => ExternalProductCatalogsTable::class]; + $this->ExternalProductCatalogs = $this->getTableLocator()->get('ExternalProductCatalogs', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ExternalProductCatalogs); + + parent::tearDown(); + } + + /** + * TestInitialize method + * + * @return void + * @uses \CakeProducts\Model\Table\ExternalProductCatalogsTable::initialize() + */ + public function testInitialize(): void + { + // verify all associations loaded + $expectedAssociations = [ + 'ProductCatalogs', + ]; + $associations = $this->ExternalProductCatalogs->associations(); + + $this->assertCount(count($expectedAssociations), $associations); + foreach ($expectedAssociations as $expectedAssociation) { + $this->assertTrue($this->ExternalProductCatalogs->hasAssociation($expectedAssociation)); + } + + // verify all behaviors loaded + $expectedBehaviors = [ + 'Timestamp', + ]; + $behaviors = $this->ExternalProductCatalogs->behaviors(); + + $this->assertCount(count($expectedBehaviors), $behaviors); + foreach ($expectedBehaviors as $expectedBehavior) { + $this->assertTrue($this->ExternalProductCatalogs->hasBehavior($expectedBehavior)); + } + } + + /** + * Test validationDefault method + * + * @return void + * @uses \CakeProducts\Model\Table\ExternalProductCatalogsTable::validationDefault() + */ + public function testValidationDefault(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } + + /** + * Test buildRules method + * + * @return void + * @uses \CakeProducts\Model\Table\ExternalProductCatalogsTable::buildRules() + */ + public function testBuildRules(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } +} diff --git a/tests/TestCase/Model/Table/ProductCatalogsTableTest.php b/tests/TestCase/Model/Table/ProductCatalogsTableTest.php new file mode 100644 index 0000000..a74882a --- /dev/null +++ b/tests/TestCase/Model/Table/ProductCatalogsTableTest.php @@ -0,0 +1,96 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Model\Table; + +use Cake\ORM\Table; +use Cake\TestSuite\TestCase; +use CakeProducts\Model\Table\ProductCatalogsTable; + +/** + * CakeProducts\Model\Table\ProductCatalogsTable Test Case + */ +class ProductCatalogsTableTest extends TestCase +{ + /** + * Test subject + * + * @var ProductCatalogsTable|Table + */ + protected $ProductCatalogs; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCatalogs', + 'plugin.CakeProducts.ProductCategories', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $config = $this->getTableLocator()->exists('ProductCatalogs') ? [] : ['className' => ProductCatalogsTable::class]; + $this->ProductCatalogs = $this->getTableLocator()->get('ProductCatalogs', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCatalogs); + + parent::tearDown(); + } + + /** + * TestInitialize method + * + * @return void + * @uses ProductCatalogsTable::initialize + */ + public function testInitialize(): void + { + // verify all associations loaded + $expectedAssociations = [ + 'ProductCategories', + 'ExternalProductCatalogs', + ]; + $associations = $this->ProductCatalogs->associations(); + + $this->assertCount(count($expectedAssociations), $associations); + foreach ($expectedAssociations as $expectedAssociation) { + $this->assertTrue($this->ProductCatalogs->hasAssociation($expectedAssociation)); + } + + // verify all behaviors loaded + $expectedBehaviors = []; + $behaviors = $this->ProductCatalogs->behaviors(); + + $this->assertCount(count($expectedBehaviors), $behaviors); + foreach ($expectedBehaviors as $expectedBehavior) { + $this->assertTrue($this->ProductCatalogs->hasBehavior($expectedBehavior)); + } + } + + /** + * Test validationDefault method + * + * @return void + * @uses ProductCatalogsTable::validationDefault + */ + public function testValidationDefault(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } +} diff --git a/tests/TestCase/Model/Table/ProductCategoriesTableTest.php b/tests/TestCase/Model/Table/ProductCategoriesTableTest.php new file mode 100644 index 0000000..79c8a19 --- /dev/null +++ b/tests/TestCase/Model/Table/ProductCategoriesTableTest.php @@ -0,0 +1,111 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Model\Table; + +use Cake\TestSuite\TestCase; +use CakeProducts\Model\Table\ProductCategoriesTable; + +/** + * CakeProducts\Model\Table\ProductCategoriesTable Test Case + */ +class ProductCategoriesTableTest extends TestCase +{ + /** + * Test subject + * + * @var \CakeProducts\Model\Table\ProductCategoriesTable + */ + protected $ProductCategories; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCategories', + 'plugin.CakeProducts.ProductCatalogs', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $config = $this->getTableLocator()->exists('ProductCategories') ? [] : ['className' => ProductCategoriesTable::class]; + $this->ProductCategories = $this->getTableLocator()->get('ProductCategories', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCategories); + + parent::tearDown(); + } + + /** + * TestInitialize method + * + * @return void + * @uses \CakeProducts\Model\Table\ProductCategoriesTable::initialize() + */ + public function testInitialize(): void + { + // verify all associations loaded + $expectedAssociations = [ + 'ProductCatalogs', + 'ParentProductCategories', + 'ChildProductCategories', +// 'Products', +// 'ProductCategoryAttributes', + ]; + $associations = $this->ProductCategories->associations(); + + $this->assertCount(count($expectedAssociations), $associations); + foreach ($expectedAssociations as $expectedAssociation) { + $this->assertTrue($this->ProductCategories->hasAssociation($expectedAssociation)); + } + + // verify all behaviors loaded + $expectedBehaviors = [ + 'Tree', + ]; + $behaviors = $this->ProductCategories->behaviors(); + + $this->assertCount(count($expectedBehaviors), $behaviors); + foreach ($expectedBehaviors as $expectedBehavior) { + $this->assertTrue($this->ProductCategories->hasBehavior($expectedBehavior)); + } + } + + /** + * Test validationDefault method + * + * @return void + * @uses \CakeProducts\Model\Table\ProductCategoriesTable::validationDefault() + */ + public function testValidationDefault(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } + + /** + * Test buildRules method + * + * @return void + * @uses \CakeProducts\Model\Table\ProductCategoriesTable::buildRules() + */ + public function testBuildRules(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } +} diff --git a/tests/TestCase/Model/Table/ProductCategoryAttributeOptionsTableTest.php b/tests/TestCase/Model/Table/ProductCategoryAttributeOptionsTableTest.php new file mode 100644 index 0000000..a5a0f53 --- /dev/null +++ b/tests/TestCase/Model/Table/ProductCategoryAttributeOptionsTableTest.php @@ -0,0 +1,104 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Model\Table; + +use Cake\TestSuite\TestCase; +use CakeProducts\Model\Table\ProductCategoryAttributeOptionsTable; + +/** + * CakeProducts\Model\Table\ProductCategoryAttributeOptionsTable Test Case + */ +class ProductCategoryAttributeOptionsTableTest extends TestCase +{ + /** + * Test subject + * + * @var ProductCategoryAttributeOptionsTable + */ + protected $ProductCategoryAttributeOptions; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCategoryAttributeOptions', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $config = $this->getTableLocator()->exists('ProductCategoryAttributeOptions') ? [] : ['className' => ProductCategoryAttributeOptionsTable::class]; + $this->ProductCategoryAttributeOptions = $this->getTableLocator()->get('ProductCategoryAttributeOptions', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCategoryAttributeOptions); + + parent::tearDown(); + } + + /** + * TestInitialize method + * + * @return void + * @uses ProductCategoryAttributeOptionsTable::initialize + */ + public function testInitialize(): void + { + // verify all associations loaded + $expectedAssociations = [ + 'ProductCategoryAttributes', + ]; + $associations = $this->ProductCategoryAttributeOptions->associations(); + + $this->assertCount(count($expectedAssociations), $associations); + foreach ($expectedAssociations as $expectedAssociation) { + $this->assertTrue($this->ProductCategoryAttributeOptions->hasAssociation($expectedAssociation)); + } + + // verify all behaviors loaded + $expectedBehaviors = []; + $behaviors = $this->ProductCategoryAttributeOptions->behaviors(); + + $this->assertCount(count($expectedBehaviors), $behaviors); + foreach ($expectedBehaviors as $expectedBehavior) { + $this->assertTrue($this->ProductCategoryAttributeOptions->hasBehavior($expectedBehavior)); + } + } + + /** + * Test validationDefault method + * + * @return void + * @uses ProductCategoryAttributeOptionsTable::validationDefault + */ + public function testValidationDefault(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } + + /** + * Test buildRules method + * + * @return void + * @uses ProductCategoryAttributeOptionsTable::buildRules + */ + public function testBuildRules(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } +} diff --git a/tests/TestCase/Model/Table/ProductCategoryAttributesTableTest.php b/tests/TestCase/Model/Table/ProductCategoryAttributesTableTest.php new file mode 100644 index 0000000..5e97ab0 --- /dev/null +++ b/tests/TestCase/Model/Table/ProductCategoryAttributesTableTest.php @@ -0,0 +1,107 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Model\Table; + +use Cake\TestSuite\TestCase; +use CakeProducts\Model\Table\ProductCategoryAttributesTable; + +/** + * CakeProducts\Model\Table\ProductCategoryAttributesTable Test Case + */ +class ProductCategoryAttributesTableTest extends TestCase +{ + /** + * Test subject + * + * @var ProductCategoryAttributesTable + */ + protected $ProductCategoryAttributes; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.ProductCategoryAttributes', + 'plugin.CakeProducts.ProductCategoryAttributeOptions', + 'plugin.CakeProducts.ProductCategories', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $config = $this->getTableLocator()->exists('ProductCategoryAttributes') ? [] : ['className' => ProductCategoryAttributesTable::class]; + $this->ProductCategoryAttributes = $this->getTableLocator()->get('ProductCategoryAttributes', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->ProductCategoryAttributes); + + parent::tearDown(); + } + + /** + * TestInitialize method + * + * @return void + * @uses ProductCategoryAttributesTable::initialize + */ + public function testInitialize(): void + { + // verify all associations loaded + $expectedAssociations = [ + 'ProductCategories', + 'ProductCategoryAttributeOptions', + ]; + $associations = $this->ProductCategoryAttributes->associations(); + + $this->assertCount(count($expectedAssociations), $associations); + foreach ($expectedAssociations as $expectedAssociation) { + $this->assertTrue($this->ProductCategoryAttributes->hasAssociation($expectedAssociation)); + } + + // verify all behaviors loaded + $expectedBehaviors = []; + $behaviors = $this->ProductCategoryAttributes->behaviors(); + + $this->assertCount(count($expectedBehaviors), $behaviors); + foreach ($expectedBehaviors as $expectedBehavior) { + $this->assertTrue($this->ProductCategoryAttributes->hasBehavior($expectedBehavior)); + } + } + + /** + * Test validationDefault method + * + * @return void + * @uses ProductCategoryAttributesTable::validationDefault + */ + public function testValidationDefault(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } + + /** + * Test buildRules method + * + * @return void + * @uses ProductCategoryAttributesTable::buildRules + */ + public function testBuildRules(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } +} diff --git a/tests/TestCase/Model/Table/ProductsTableTest.php b/tests/TestCase/Model/Table/ProductsTableTest.php new file mode 100644 index 0000000..57bb0bd --- /dev/null +++ b/tests/TestCase/Model/Table/ProductsTableTest.php @@ -0,0 +1,105 @@ +<?php +declare(strict_types=1); + +namespace CakeProducts\Test\TestCase\Model\Table; + +use Cake\TestSuite\TestCase; +use CakeProducts\Model\Table\ProductsTable; + +/** + * CakeProducts\Model\Table\ProductsTable Test Case + */ +class ProductsTableTest extends TestCase +{ + /** + * Test subject + * + * @var ProductsTable + */ + protected $Products; + + /** + * Fixtures + * + * @var array<string> + */ + protected array $fixtures = [ + 'plugin.CakeProducts.Products', + 'plugin.CakeProducts.ProductCategories', + ]; + + /** + * setUp method + * + * @return void + */ + protected function setUp(): void + { + parent::setUp(); + $config = $this->getTableLocator()->exists('Products') ? [] : ['className' => ProductsTable::class]; + $this->Products = $this->getTableLocator()->get('Products', $config); + } + + /** + * tearDown method + * + * @return void + */ + protected function tearDown(): void + { + unset($this->Products); + + parent::tearDown(); + } + + /** + * TestInitialize method + * + * @return void + * @uses ProductsTable::initialize + */ + public function testInitialize(): void + { + // verify all associations loaded + $expectedAssociations = [ + 'ProductCategories', + ]; + $associations = $this->Products->associations(); + + $this->assertCount(count($expectedAssociations), $associations); + foreach ($expectedAssociations as $expectedAssociation) { + $this->assertTrue($this->Products->hasAssociation($expectedAssociation)); + } + + // verify all behaviors loaded + $expectedBehaviors = []; + $behaviors = $this->Products->behaviors(); + + $this->assertCount(count($expectedBehaviors), $behaviors); + foreach ($expectedBehaviors as $expectedBehavior) { + $this->assertTrue($this->Products->hasBehavior($expectedBehavior)); + } + } + + /** + * Test validationDefault method + * + * @return void + * @uses ProductsTable::validationDefault + */ + public function testValidationDefault(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } + + /** + * Test buildRules method + * + * @return void + * @uses ProductsTable::buildRules + */ + public function testBuildRules(): void + { + $this->markTestIncomplete('Not implemented yet.'); + } +} diff --git a/tests/bootstrap.php b/tests/bootstrap.php new file mode 100644 index 0000000..5ae28bc --- /dev/null +++ b/tests/bootstrap.php @@ -0,0 +1,55 @@ +<?php +declare(strict_types=1); + +/** + * Test suite bootstrap for CakeProducts. + * + * This function is used to find the location of CakePHP whether CakePHP + * has been installed as a dependency of the plugin, or the plugin is itself + * installed as a dependency of an application. + */ +$findRoot = function ($root) { + do { + $lastRoot = $root; + $root = dirname($root); + if (is_dir($root . '/vendor/cakephp/cakephp')) { + return $root; + } + } while ($root !== $lastRoot); + + throw new Exception('Cannot find the root of the application, unable to run tests'); +}; +$root = $findRoot(__FILE__); +unset($findRoot); + +chdir($root); + +require_once $root . '/vendor/autoload.php'; + +/** + * Define fallback values for required constants and configuration. + * To customize constants and configuration remove this require + * and define the data required by your plugin here. + */ +require_once $root . '/vendor/cakephp/cakephp/tests/bootstrap.php'; + +if (file_exists($root . '/config/bootstrap.php')) { + require $root . '/config/bootstrap.php'; + + return; +} + +/** + * Load schema from a SQL dump file. + * + * If your plugin does not use database fixtures you can + * safely delete this. + * + * If you want to support multiple databases, consider + * using migrations to provide schema for your plugin, + * and using \Migrations\TestSuite\Migrator to load schema. + */ +use Cake\TestSuite\Fixture\SchemaLoader; + +// Load a schema dump file. +(new SchemaLoader())->loadSqlFiles('tests/schema.sql', 'test'); diff --git a/tests/schema.sql b/tests/schema.sql new file mode 100644 index 0000000..ef9d09a --- /dev/null +++ b/tests/schema.sql @@ -0,0 +1 @@ +-- Test database schema for CakeProducts diff --git a/webroot/.gitkeep b/webroot/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/webroot/js/product_category_attribute_options.js b/webroot/js/product_category_attribute_options.js new file mode 100644 index 0000000..4404709 --- /dev/null +++ b/webroot/js/product_category_attribute_options.js @@ -0,0 +1,15 @@ +const addOptionButton = document.getElementById('add-option-button'); +const attributeOptionPrefixInput = document.getElementById('attribute_options_prefix'); +if (addOptionButton && attributeOptionPrefixInput) { + addOptionButton.addEventListener('click', addOptionButtonClicked); +} +function addOptionButtonClicked(e) +{ + e.preventDefault(); + console.debug('attributeOptionPrefixInput.value'); + console.debug(attributeOptionPrefixInput.value); + attributeOptionPrefixInput.value = parseInt(attributeOptionPrefixInput.value) + 1; + attributeOptionPrefixInput.dispatchEvent(new Event('change')); + console.debug('attributeOptionPrefixInput.value'); + console.debug(attributeOptionPrefixInput.value); +}