+
+
+
+ = $this->Form->create($externalProductCatalog) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+
+
+
+
+
+ = $this->Form->create($externalProductCatalog) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+ $externalProductCatalogs
+ */
+?>
+
+ = $this->Html->link(__('New External Product Catalog'), ['action' => 'add'], ['class' => 'button float-right']) ?>
+
= __('External Product Catalogs') ?>
+
+
+
+
+ = $this->Paginator->sort('id') ?> |
+ = $this->Paginator->sort('product_catalog_id') ?> |
+ = $this->Paginator->sort('base_url') ?> |
+ = $this->Paginator->sort('api_url') ?> |
+ = $this->Paginator->sort('created') ?> |
+ = $this->Paginator->sort('deleted') ?> |
+ = $this->Paginator->sort('enabled') ?> |
+ = __('Actions') ?> |
+
+
+
+
+
+ = $this->Number->format($externalProductCatalog->id) ?> |
+ = $externalProductCatalog->hasValue('product_catalog') ? $this->Html->link($externalProductCatalog->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $externalProductCatalog->product_catalog->id]) : '' ?> |
+ = h($externalProductCatalog->base_url) ?> |
+ = h($externalProductCatalog->api_url) ?> |
+ = h($externalProductCatalog->created) ?> |
+ = h($externalProductCatalog->deleted) ?> |
+ = h($externalProductCatalog->enabled) ?> |
+
+ = $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)]) ?>
+ |
+
+
+
+
+
+
+
+
= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?>
+
+
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 @@
+
+
+
+
+
+
= h($externalProductCatalog->base_url) ?>
+
+
+ = __('Product Catalog') ?> |
+ = $externalProductCatalog->hasValue('product_catalog') ? $this->Html->link($externalProductCatalog->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $externalProductCatalog->product_catalog->id]) : '' ?> |
+
+
+ = __('Base Url') ?> |
+ = h($externalProductCatalog->base_url) ?> |
+
+
+ = __('Api Url') ?> |
+ = h($externalProductCatalog->api_url) ?> |
+
+
+ = __('Id') ?> |
+ = $this->Number->format($externalProductCatalog->id) ?> |
+
+
+ = __('Created') ?> |
+ = h($externalProductCatalog->created) ?> |
+
+
+ = __('Deleted') ?> |
+ = h($externalProductCatalog->deleted) ?> |
+
+
+ = __('Enabled') ?> |
+ = $externalProductCatalog->enabled ? __('Yes') : __('No'); ?> |
+
+
+
+
+
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 @@
+
+
+
+
+
+ = $this->Form->create($productCatalog) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+
+
+
+
+
+ = $this->Form->create($productCatalog) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+ $productCatalogs
+ */
+?>
+
+ = $this->Html->link(__('New Product Catalog'), ['action' => 'add'], ['class' => 'button float-right']) ?>
+
= __('Product Catalogs') ?>
+
+
+
+
+ = $this->Paginator->sort('id') ?> |
+ = $this->Paginator->sort('name') ?> |
+ = $this->Paginator->sort('catalog_description') ?> |
+ = $this->Paginator->sort('enabled') ?> |
+ = __('Actions') ?> |
+
+
+
+
+
+ = $productCatalog->id; ?> |
+ = h($productCatalog->name) ?> |
+ = h($productCatalog->catalog_description) ?> |
+ = h($productCatalog->enabled) ?> |
+
+ = $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)]) ?>
+ |
+
+
+
+
+
+
+
+
= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?>
+
+
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 @@
+
+
+
+
+
+
= h($productCatalog->name) ?>
+
+
+ = __('Name') ?> |
+ = h($productCatalog->name) ?> |
+
+
+ = __('Catalog Description') ?> |
+ = h($productCatalog->catalog_description) ?> |
+
+
+ = __('Id') ?> |
+ = $productCatalog->id; ?> |
+
+
+ = __('Enabled') ?> |
+ = $productCatalog->enabled ? __('Yes') : __('No'); ?> |
+
+
+
+
+
+
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 @@
+
+
+
+
+
+ = $this->Form->create($productCategory) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+
+
+
+
+
+ = $this->Form->create($productCategory) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+ $productCategories
+ */
+?>
+
+ = $this->Html->link(__('New Product Category'), ['action' => 'add'], ['class' => 'button float-right']) ?>
+
= __('Product Categories') ?>
+
+
+
+
+ = $this->Paginator->sort('id') ?> |
+ = $this->Paginator->sort('product_catalog_id') ?> |
+ = $this->Paginator->sort('name') ?> |
+ = $this->Paginator->sort('parent_id') ?> |
+ = $this->Paginator->sort('enabled') ?> |
+ = __('Actions') ?> |
+
+
+
+
+
+ = $productCategory->id; ?> |
+ = $productCategory->hasValue('product_catalog') ? $this->Html->link($productCategory->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $productCategory->product_catalog->id]) : '' ?> |
+ = h($productCategory->name) ?> |
+ = $productCategory->hasValue('parent_product_category') ? $this->Html->link($productCategory->parent_product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategory->parent_product_category->id]) : '' ?> |
+ = h($productCategory->enabled) ?> |
+
+ = $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)]) ?>
+ |
+
+
+
+
+
+
+
+
= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?>
+
+
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 @@
+
+
+
+
+
+
= h($productCategory->name) ?>
+
+
+ = __('Product Catalog') ?> |
+ = $productCategory->hasValue('product_catalog') ? $this->Html->link($productCategory->product_catalog->name, ['controller' => 'ProductCatalogs', 'action' => 'view', $productCategory->product_catalog->id]) : '' ?> |
+
+
+ = __('Name') ?> |
+ = h($productCategory->name) ?> |
+
+
+ = __('Parent Product Category') ?> |
+ = $productCategory->hasValue('parent_product_category') ? $this->Html->link($productCategory->parent_product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategory->parent_product_category->id]) : '' ?> |
+
+
+ = __('Id') ?> |
+ = $productCategory->id; ?> |
+
+
+ = __('Lft') ?> |
+ = $this->Number->format($productCategory->lft) ?> |
+
+
+ = __('Rght') ?> |
+ = $this->Number->format($productCategory->rght) ?> |
+
+
+ = __('Enabled') ?> |
+ = $productCategory->enabled ? __('Yes') : __('No'); ?> |
+
+
+
+
= __('Category Description') ?>
+
+ = $this->Text->autoParagraph(h($productCategory->category_description)); ?>
+
+
+
+
+
+
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 @@
+setLayout('ajax');
+$prefix = $prefix ?? '';
+if ($this->request->getQuery('prefix') !== null) {
+ $prefix = 'product_category_attribute_options.' . $this->request->getQuery('prefix') . '.';
+}
+echo '
+
+
+
+ = $this->Form->create($productCategoryAttribute) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
+= $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 @@
+ $productCategoryAttributes
+ */
+?>
+
+ = $this->Html->link(__('New Product Category Attribute'), ['action' => 'add'], ['class' => 'button float-right']) ?>
+
= __('Product Category Attributes') ?>
+
+
+
+
+ = $this->Paginator->sort('id') ?> |
+ = $this->Paginator->sort('name') ?> |
+ = $this->Paginator->sort('product_category_id') ?> |
+ = $this->Paginator->sort('attribute_type_id') ?> |
+ = $this->Paginator->sort('enabled') ?> |
+ = __('Actions') ?> |
+
+
+
+
+
+ = $productCategoryAttribute->id ?> |
+ = h($productCategoryAttribute->name) ?> |
+ = $productCategoryAttribute->hasValue('product_category') ? $this->Html->link($productCategoryAttribute->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategoryAttribute->product_category->id]) : '' ?> |
+ = $productCategoryAttribute->attribute_type_id->name ?> |
+ = h($productCategoryAttribute->enabled) ?> |
+
+ = $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)]) ?>
+ |
+
+
+
+
+
+
+
+
= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?>
+
+
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 @@
+
+
+
+
+
+
= h($productCategoryAttribute->name) ?>
+
+
+ = __('Name') ?> |
+ = h($productCategoryAttribute->name) ?> |
+
+
+ = __('Product Category') ?> |
+ = $productCategoryAttribute->hasValue('product_category') ? $this->Html->link($productCategoryAttribute->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $productCategoryAttribute->product_category->id]) : '' ?> |
+
+
+ = __('Id') ?> |
+ = $productCategoryAttribute->id ?> |
+
+
+ = __('Enabled') ?> |
+ = $productCategoryAttribute->enabled ? __('Yes') : __('No'); ?> |
+
+
+
+
+
+
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 @@
+
+
+
+
+
+ = $this->Form->create($product) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+
+
+
+
+
+ = $this->Form->create($product) ?>
+
+ = $this->Form->button(__('Submit')) ?>
+ = $this->Form->end() ?>
+
+
+
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 @@
+ $products
+ */
+?>
+
+ = $this->Html->link(__('New Product'), ['action' => 'add'], ['class' => 'button float-right']) ?>
+
= __('Products') ?>
+
+
+
+
+ = $this->Paginator->sort('id') ?> |
+ = $this->Paginator->sort('name') ?> |
+ = $this->Paginator->sort('product_category_id') ?> |
+ = $this->Paginator->sort('product_type_id') ?> |
+ = __('Actions') ?> |
+
+
+
+
+
+ = $this->Number->format($product->id) ?> |
+ = h($product->name) ?> |
+ = $product->hasValue('product_category') ? $this->Html->link($product->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $product->product_category->id]) : '' ?> |
+ = $product->product_type_id->name ?> |
+
+ = $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)]) ?>
+ |
+
+
+
+
+
+
+
+
= $this->Paginator->counter(__('Page {{page}} of {{pages}}, showing {{current}} record(s) out of {{count}} total')) ?>
+
+
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 @@
+
+
+
+
+
+
= h($product->name) ?>
+
+
+ = __('Name') ?> |
+ = h($product->name) ?> |
+
+
+ = __('Product Category') ?> |
+ = $product->hasValue('product_category') ? $this->Html->link($product->product_category->name, ['controller' => 'ProductCategories', 'action' => 'view', $product->product_category->id]) : '' ?> |
+
+
+ = __('Id') ?> |
+ = $this->Number->format($product->id) ?> |
+
+
+ = __('Product Type Id') ?> |
+ = $product->product_type_id->name ?> |
+
+
+
+
+
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 @@
+hasValue('product_category_attribute_options') || empty($productCategoryAttribute->product_category_attribute_options) ? 0 : count($productCategoryAttribute->product_category_attribute_options);
+$cnt = 0;
+$prefix = $prefix ?? '';
+?>
+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');
+?>
+
+ hasValue('product_category_attribute_options')) : ?>
+ product_category_attribute_options as $attributeOption) {
+ $prefix = 'product_category_attribute_options.' . $cnt . '.';
+ echo '
';
+ echo $this->element('CakeProducts.ProductCategoryAttributes/product_category_attribute_option_form', [
+ 'attributeOption' => $attributeOption,
+ 'prefix' => $prefix,
+ ]);
+ $cnt++;
+ } ?>
+
+
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 @@
+
+
+
+
+ = $this->Form->control($prefix . 'attribute_value', [
+ 'label' => 'Value',
+ ]); ?>
+
+
+ = $this->Form->control($prefix . 'attribute_label', [
+ 'label' => 'Label',
+ ]); ?>
+
+
+ = $this->Form->control($prefix . 'enabled', [
+ 'type' => 'checkbox',
+ 'checked' => true,
+ 'label' => 'Enabled',
+ ]); ?>
+
+
+
+
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 @@
+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 @@
+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 @@
+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 @@
+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 @@
+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 @@
+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 @@
+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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+
+ */
+ 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 @@
+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);
+}