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);
+}