Merge branch '0.x' into prod
This commit is contained in:
133
.gitea/workflows/ci.yaml
Normal file
133
.gitea/workflows/ci.yaml
Normal file
@@ -0,0 +1,133 @@
|
|||||||
|
name: CI
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
pull_request:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
testsuite:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
php-version: ['8.2', '8.4']
|
||||||
|
# db-type: ['mysql']
|
||||||
|
db-type: ['sqlite', 'mysql', 'pgsql']
|
||||||
|
prefer-lowest: ['']
|
||||||
|
include:
|
||||||
|
- php-version: '8.2'
|
||||||
|
db-type: 'sqlite'
|
||||||
|
prefer-lowest: 'prefer-lowest'
|
||||||
|
|
||||||
|
services:
|
||||||
|
mysql8:
|
||||||
|
image: mysql:8.0
|
||||||
|
env:
|
||||||
|
MYSQL_ALLOW_EMPTY_PASSWORD: yes
|
||||||
|
MYSQL_DATABASE: cakephp
|
||||||
|
# services:
|
||||||
|
# postgres:
|
||||||
|
# image: postgres
|
||||||
|
# ports:
|
||||||
|
# - 5432:5432
|
||||||
|
# env:
|
||||||
|
# POSTGRES_PASSWORD: postgres
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Setup PHP
|
||||||
|
uses: shivammathur/setup-php@v2
|
||||||
|
with:
|
||||||
|
php-version: ${{ matrix.php-version }}
|
||||||
|
extensions: mbstring, intl, sqlite, pdo_${{ matrix.db-type }}
|
||||||
|
coverage: pcov
|
||||||
|
|
||||||
|
- name: Get composer cache directory
|
||||||
|
id: composercache
|
||||||
|
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Cache dependencies
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ steps.composercache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-composer-${{ steps.key-date.outputs.date }}-${{ hashFiles('composer.json') }}-${{ matrix.prefer-lowest }}
|
||||||
|
|
||||||
|
- name: Composer install
|
||||||
|
run: |
|
||||||
|
composer --version
|
||||||
|
if ${{ matrix.prefer-lowest == 'prefer-lowest' }}
|
||||||
|
then
|
||||||
|
composer update --prefer-lowest --prefer-stable
|
||||||
|
composer require --dev dereuromark/composer-prefer-lowest:dev-master
|
||||||
|
else
|
||||||
|
composer install --no-progress --prefer-dist --optimize-autoloader
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Setup problem matchers for PHPUnit
|
||||||
|
if: matrix.db-type == 'mysql'
|
||||||
|
run: echo "::add-matcher::${{ runner.tool_cache }}/phpunit.json"
|
||||||
|
|
||||||
|
|
||||||
|
- name: Run PHPUnit tests
|
||||||
|
env:
|
||||||
|
TEST_MYSQL_HOST: mysql8
|
||||||
|
TEST_MYSQL_CHARSET: utf8mb4
|
||||||
|
TEST_MYSQL_DBNAME: cakephp
|
||||||
|
TEST_MYSQL_USERNAME: root
|
||||||
|
TEST_MYSQL_PASSWORD:
|
||||||
|
run: |
|
||||||
|
if [[ ${{ matrix.php-version }} == '8.2' ]]; then
|
||||||
|
vendor/bin/phpunit --coverage-clover=coverage.xml
|
||||||
|
else
|
||||||
|
vendor/bin/phpunit
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Validate prefer-lowest
|
||||||
|
if: matrix.prefer-lowest == 'prefer-lowest'
|
||||||
|
run: vendor/bin/validate-prefer-lowest -m
|
||||||
|
|
||||||
|
# - name: Upload coverage reports to Codecov
|
||||||
|
# if: success() && matrix.php-version == '8.2'
|
||||||
|
# uses: codecov/codecov-action@v4
|
||||||
|
# with:
|
||||||
|
# token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
|
|
||||||
|
validation:
|
||||||
|
name: Coding Standard & Static Analysis
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Setup PHP
|
||||||
|
uses: shivammathur/setup-php@v2
|
||||||
|
with:
|
||||||
|
php-version: '8.2'
|
||||||
|
extensions: mbstring, intl, sqlite
|
||||||
|
coverage: none
|
||||||
|
|
||||||
|
- name: Get composer cache directory
|
||||||
|
id: composercache
|
||||||
|
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Cache dependencies
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ steps.composercache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-composer-${{ steps.key-date.outputs.date }}-${{ hashFiles('composer.json') }}-${{ matrix.prefer-lowest }}
|
||||||
|
|
||||||
|
- name: Composer install
|
||||||
|
run: |
|
||||||
|
composer --version
|
||||||
|
if ${{ matrix.prefer-lowest == 'prefer-lowest' }}
|
||||||
|
then
|
||||||
|
composer update --prefer-lowest --prefer-stable
|
||||||
|
composer require --dev dereuromark/composer-prefer-lowest:dev-master
|
||||||
|
else
|
||||||
|
composer install --no-progress --prefer-dist --optimize-autoloader
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Run phpstan
|
||||||
|
run: vendor/bin/phpstan analyse src --error-format=github
|
||||||
|
|
||||||
|
- name: Run phpcs
|
||||||
|
run: composer cs-check
|
||||||
17
.gitea/workflows/release.yaml
Normal file
17
.gitea/workflows/release.yaml
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
name: Release
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- "v*.*.*"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- name: Release
|
||||||
|
uses: softprops/action-gh-release@v2
|
||||||
|
|
||||||
|
|
||||||
31
.gitea/workflows/status.yaml
Normal file
31
.gitea/workflows/status.yaml
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
---
|
||||||
|
name: CI Build Status
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_run:
|
||||||
|
workflows:
|
||||||
|
- CI
|
||||||
|
types:
|
||||||
|
- completed
|
||||||
|
jobs:
|
||||||
|
on-success:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.event.workflow_run.conclusion == 'success' }}
|
||||||
|
steps:
|
||||||
|
- name: Webhook
|
||||||
|
uses: zzzze/webhook-trigger@master
|
||||||
|
with:
|
||||||
|
data: "{\"text\":\"${{ gitea.repository }} build succeeded 🎉\"}"
|
||||||
|
webhook_url: ${{ secrets.BUILD_STATUS_NOTIFY_WEBHOOK_URL }}
|
||||||
|
options: "-H \"Content-Type: application/json\""
|
||||||
|
|
||||||
|
on-failure:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.event.workflow_run.conclusion == 'failure' }}
|
||||||
|
steps:
|
||||||
|
- name: Webhook
|
||||||
|
uses: zzzze/webhook-trigger@master
|
||||||
|
with:
|
||||||
|
data: "{\"text\":\"${{ gitea.repository }} build failed 🔎\"}"
|
||||||
|
webhook_url: ${{ secrets.BUILD_STATUS_NOTIFY_WEBHOOK_URL }}
|
||||||
|
options: "-H \"Content-Type: application/json\""
|
||||||
127
.github/workflows/ci.yml
vendored
Normal file
127
.github/workflows/ci.yml
vendored
Normal file
@@ -0,0 +1,127 @@
|
|||||||
|
name: CI
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
pull_request:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
testsuite:
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
php-version: ['8.1', '8.4']
|
||||||
|
db-type: ['sqlite', 'mysql', 'pgsql']
|
||||||
|
prefer-lowest: ['']
|
||||||
|
include:
|
||||||
|
- php-version: '8.1'
|
||||||
|
db-type: 'sqlite'
|
||||||
|
prefer-lowest: 'prefer-lowest'
|
||||||
|
|
||||||
|
services:
|
||||||
|
postgres:
|
||||||
|
image: postgres
|
||||||
|
ports:
|
||||||
|
- 5432:5432
|
||||||
|
env:
|
||||||
|
POSTGRES_PASSWORD: postgres
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup Service
|
||||||
|
if: matrix.db-type == 'mysql'
|
||||||
|
run: |
|
||||||
|
sudo service mysql start
|
||||||
|
mysql -h 127.0.0.1 -u root -proot -e 'CREATE DATABASE cakephp;'
|
||||||
|
|
||||||
|
- name: Setup PHP
|
||||||
|
uses: shivammathur/setup-php@v2
|
||||||
|
with:
|
||||||
|
php-version: ${{ matrix.php-version }}
|
||||||
|
extensions: mbstring, intl, bcmath, pdo_${{ matrix.db-type }}
|
||||||
|
coverage: pcov
|
||||||
|
|
||||||
|
- name: Get composer cache directory
|
||||||
|
id: composercache
|
||||||
|
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Cache dependencies
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ steps.composercache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-composer-${{ steps.key-date.outputs.date }}-${{ hashFiles('composer.json') }}-${{ matrix.prefer-lowest }}
|
||||||
|
|
||||||
|
- name: Composer install
|
||||||
|
run: |
|
||||||
|
composer --version
|
||||||
|
if ${{ matrix.prefer-lowest == 'prefer-lowest' }}
|
||||||
|
then
|
||||||
|
composer update --prefer-lowest --prefer-stable
|
||||||
|
composer require --dev dereuromark/composer-prefer-lowest:dev-master
|
||||||
|
else
|
||||||
|
composer install --no-progress --prefer-dist --optimize-autoloader
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Setup problem matchers for PHPUnit
|
||||||
|
if: matrix.db-type == 'mysql'
|
||||||
|
run: echo "::add-matcher::${{ runner.tool_cache }}/phpunit.json"
|
||||||
|
|
||||||
|
- name: Wait for MySQL
|
||||||
|
if: matrix.db-type == 'mysql'
|
||||||
|
run: while ! `mysqladmin ping -h 127.0.0.1 --silent`; do printf 'Waiting for MySQL...\n'; sleep 2; done;
|
||||||
|
|
||||||
|
- name: Run PHPUnit
|
||||||
|
run: |
|
||||||
|
if [[ ${{ matrix.db-type }} == 'sqlite' ]]; then export DB_URL='sqlite:///:memory:'; fi
|
||||||
|
if [[ ${{ matrix.db-type }} == 'mysql' ]]; then export DB_URL='mysql://root:root@127.0.0.1/cakephp?encoding=utf8'; fi
|
||||||
|
if [[ ${{ matrix.db-type }} == 'pgsql' ]]; then export DB_URL='postgres://postgres:postgres@127.0.0.1/postgres'; fi
|
||||||
|
if [[ ${{ matrix.php-version }} == '8.1' ]]; then
|
||||||
|
vendor/bin/phpunit --coverage-clover=coverage.xml
|
||||||
|
else
|
||||||
|
vendor/bin/phpunit
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Validate prefer-lowest
|
||||||
|
if: matrix.prefer-lowest == 'prefer-lowest'
|
||||||
|
run: vendor/bin/validate-prefer-lowest -m
|
||||||
|
|
||||||
|
# - name: Upload coverage reports to Codecov
|
||||||
|
# if: success() && matrix.php-version == '8.1'
|
||||||
|
# uses: codecov/codecov-action@v4
|
||||||
|
# with:
|
||||||
|
# token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
|
|
||||||
|
validation:
|
||||||
|
name: Coding Standard & Static Analysis
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup PHP
|
||||||
|
uses: shivammathur/setup-php@v2
|
||||||
|
with:
|
||||||
|
php-version: '8.1'
|
||||||
|
extensions: mbstring, intl
|
||||||
|
coverage: none
|
||||||
|
|
||||||
|
- name: Get composer cache directory
|
||||||
|
id: composercache
|
||||||
|
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Cache dependencies
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ steps.composercache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-composer-${{ steps.key-date.outputs.date }}-${{ hashFiles('composer.json') }}-${{ matrix.prefer-lowest }}
|
||||||
|
|
||||||
|
- name: Composer phpstan setup
|
||||||
|
run: composer stan-setup
|
||||||
|
|
||||||
|
- name: Run phpstan
|
||||||
|
run: vendor/bin/phpstan analyse --error-format=github
|
||||||
|
|
||||||
|
- name: Run phpcs
|
||||||
|
run: composer cs-check
|
||||||
@@ -7,6 +7,6 @@ You can install this plugin into your CakePHP application using [composer](https
|
|||||||
The recommended way to install composer packages is:
|
The recommended way to install composer packages is:
|
||||||
|
|
||||||
```
|
```
|
||||||
composer require hi-powereddev/cake-contact-us
|
composer require hi-powered-dev/cake-contact-us
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
"type": "cakephp-plugin",
|
"type": "cakephp-plugin",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"require": {
|
"require": {
|
||||||
"php": ">=8.1",
|
"php": ">=8.2",
|
||||||
"cakephp/cakephp": "^5.0.1"
|
"cakephp/cakephp": "^5.0.1"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
|
|||||||
@@ -28,17 +28,22 @@ class CreateContactUsFormSubmissions extends AbstractMigration {
|
|||||||
'limit' => 45,
|
'limit' => 45,
|
||||||
'null' => true,
|
'null' => true,
|
||||||
]);
|
]);
|
||||||
$table->addColumn('name', 'string', [
|
$table->addColumn('first_name', 'string', [
|
||||||
'default' => null,
|
'default' => null,
|
||||||
'limit' => 255,
|
'limit' => 255,
|
||||||
'null' => false,
|
'null' => true,
|
||||||
|
]);
|
||||||
|
$table->addColumn('last_name', 'string', [
|
||||||
|
'default' => null,
|
||||||
|
'limit' => 255,
|
||||||
|
'null' => true,
|
||||||
]);
|
]);
|
||||||
$table->addColumn('email', 'string', [
|
$table->addColumn('email', 'string', [
|
||||||
'default' => null,
|
'default' => null,
|
||||||
'limit' => 255,
|
'limit' => 255,
|
||||||
'null' => true,
|
'null' => true,
|
||||||
]);
|
]);
|
||||||
$table->addColumn('subject', 'string', [
|
$table->addColumn('contact_subject', 'string', [
|
||||||
'default' => null,
|
'default' => null,
|
||||||
'limit' => 255,
|
'limit' => 255,
|
||||||
'null' => true,
|
'null' => true,
|
||||||
|
|||||||
@@ -8,11 +8,12 @@
|
|||||||
<php>
|
<php>
|
||||||
<ini name="memory_limit" value="-1"/>
|
<ini name="memory_limit" value="-1"/>
|
||||||
<ini name="apc.enable_cli" value="1"/>
|
<ini name="apc.enable_cli" value="1"/>
|
||||||
|
<env name="FIXTURE_SCHEMA_METADATA" value="tests/schema.php"/>
|
||||||
</php>
|
</php>
|
||||||
|
|
||||||
<!-- Add any additional test suites you want to run here -->
|
<!-- Add any additional test suites you want to run here -->
|
||||||
<testsuites>
|
<testsuites>
|
||||||
<testsuite name="CakeContactUs">
|
<testsuite name="CakeConctactUs">
|
||||||
<directory>tests/TestCase/</directory>
|
<directory>tests/TestCase/</directory>
|
||||||
</testsuite>
|
</testsuite>
|
||||||
</testsuites>
|
</testsuites>
|
||||||
|
|||||||
@@ -90,7 +90,6 @@ class ContactUsComponent extends Component {
|
|||||||
'contactUsFormSubmission' => $contactUsFormSubmission,
|
'contactUsFormSubmission' => $contactUsFormSubmission,
|
||||||
], $this->getController());
|
], $this->getController());
|
||||||
$result = $event->getResult();
|
$result = $event->getResult();
|
||||||
|
|
||||||
Log::debug(print_r('$result', true));
|
Log::debug(print_r('$result', true));
|
||||||
Log::debug(print_r($result, true));
|
Log::debug(print_r($result, true));
|
||||||
if ($result instanceof EntityInterface) {
|
if ($result instanceof EntityInterface) {
|
||||||
@@ -104,15 +103,13 @@ class ContactUsComponent extends Component {
|
|||||||
if ($contactUsFormSubmissionSaved) {
|
if ($contactUsFormSubmissionSaved) {
|
||||||
return $this->_afterFormSaved($contactUsFormSubmissionSaved);
|
return $this->_afterFormSaved($contactUsFormSubmissionSaved);
|
||||||
}
|
}
|
||||||
|
|
||||||
// @TODO contact us form submission failed - handle here
|
// @TODO contact us form submission failed - handle here
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($event->isStopped()) {
|
if ($event->isStopped()) {
|
||||||
return $this->getController()->redirect($event->getResult());
|
return $this->getController()->redirect($event->getResult());
|
||||||
}
|
}
|
||||||
|
if (!$postData) {
|
||||||
if (!$this->getController()->getRequest()->is('post')) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -45,7 +45,7 @@ class ContactUsFormSubmissionsMailer extends Mailer {
|
|||||||
*/
|
*/
|
||||||
protected function backend(ContactUsFormSubmission $contactUsFormSubmission, array $options = []) {
|
protected function backend(ContactUsFormSubmission $contactUsFormSubmission, array $options = []) {
|
||||||
$subject = __d('cake_contact_us', 'Contact Us Form Submitted');
|
$subject = __d('cake_contact_us', 'Contact Us Form Submitted');
|
||||||
$name = isset($contactUsFormSubmission->name) ? ' by ' . $contactUsFormSubmission->name : '';
|
$name = isset($contactUsFormSubmission->full_name) ? ' by ' . $contactUsFormSubmission->full_name : '';
|
||||||
|
|
||||||
$to = Configure::readOrFail('ContactUs.email.backend.to');
|
$to = Configure::readOrFail('ContactUs.email.backend.to');
|
||||||
$cc = Configure::read('ContactUs.email.backend.cc', []);
|
$cc = Configure::read('ContactUs.email.backend.cc', []);
|
||||||
@@ -62,7 +62,7 @@ class ContactUsFormSubmissionsMailer extends Mailer {
|
|||||||
|
|
||||||
if ($cc) {
|
if ($cc) {
|
||||||
$cc = !is_array($cc) ? [$cc] : $cc;
|
$cc = !is_array($cc) ? [$cc] : $cc;
|
||||||
$this->setCc($to);
|
$this->setCc($cc);
|
||||||
}
|
}
|
||||||
if ($bcc) {
|
if ($bcc) {
|
||||||
$bcc = !is_array($bcc) ? [$bcc] : $bcc;
|
$bcc = !is_array($bcc) ? [$bcc] : $bcc;
|
||||||
|
|||||||
@@ -11,9 +11,10 @@ use Cake\ORM\Entity;
|
|||||||
* @property string $id
|
* @property string $id
|
||||||
* @property \Cake\I18n\DateTime $submitted_at
|
* @property \Cake\I18n\DateTime $submitted_at
|
||||||
* @property string $client_ip
|
* @property string $client_ip
|
||||||
* @property string $name
|
* @property string|null $first_name
|
||||||
|
* @property string|null $last_name
|
||||||
* @property string|null $email
|
* @property string|null $email
|
||||||
* @property string|null $subject
|
* @property string|null $contact_subject
|
||||||
* @property string $message
|
* @property string $message
|
||||||
* @property \Cake\I18n\DateTime|null $confirm_email_sent
|
* @property \Cake\I18n\DateTime|null $confirm_email_sent
|
||||||
* @property \Cake\I18n\DateTime|null $backend_email_sent
|
* @property \Cake\I18n\DateTime|null $backend_email_sent
|
||||||
@@ -32,12 +33,20 @@ class ContactUsFormSubmission extends Entity {
|
|||||||
protected array $_accessible = [
|
protected array $_accessible = [
|
||||||
'submitted_at' => true,
|
'submitted_at' => true,
|
||||||
'client_ip' => true,
|
'client_ip' => true,
|
||||||
'name' => true,
|
'first_name' => true,
|
||||||
|
'last_name' => true,
|
||||||
'email' => true,
|
'email' => true,
|
||||||
'subject' => true,
|
'contact_subject' => true,
|
||||||
'message' => true,
|
'message' => true,
|
||||||
'confirm_email_sent' => true,
|
'confirm_email_sent' => true,
|
||||||
'backend_email_sent' => true,
|
'backend_email_sent' => true,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
protected function _getFullName(): string {
|
||||||
|
return $this->get('first_name') . ' ' . $this->get('last_name');
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ class ContactUsFormSubmissionsTable extends Table {
|
|||||||
parent::initialize($config);
|
parent::initialize($config);
|
||||||
|
|
||||||
$this->setTable('contact_us_form_submissions');
|
$this->setTable('contact_us_form_submissions');
|
||||||
$this->setDisplayField('name');
|
$this->setDisplayField('email');
|
||||||
$this->setPrimaryKey('id');
|
$this->setPrimaryKey('id');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,10 +69,15 @@ class ContactUsFormSubmissionsTable extends Table {
|
|||||||
->allowEmptyString('client_ip');
|
->allowEmptyString('client_ip');
|
||||||
|
|
||||||
$validator
|
$validator
|
||||||
->scalar('name')
|
->scalar('first_name')
|
||||||
->maxLength('name', 255)
|
->maxLength('first_name', 255)
|
||||||
->requirePresence('name', 'create')
|
->requirePresence('first_name', 'create')
|
||||||
->notEmptyString('name');
|
->notEmptyString('first_name');
|
||||||
|
|
||||||
|
$validator
|
||||||
|
->scalar('last_name')
|
||||||
|
->maxLength('last_name', 255)
|
||||||
|
->allowEmptyString('last_name');
|
||||||
|
|
||||||
// email
|
// email
|
||||||
$validator->email('email');
|
$validator->email('email');
|
||||||
@@ -82,14 +87,14 @@ class ContactUsFormSubmissionsTable extends Table {
|
|||||||
$validator->allowEmptyString('email');
|
$validator->allowEmptyString('email');
|
||||||
}
|
}
|
||||||
|
|
||||||
// subject
|
// contact_subject
|
||||||
$validator
|
$validator
|
||||||
->scalar('subject')
|
->scalar('contact_subject')
|
||||||
->maxLength('subject', 255);
|
->maxLength('contact_subject', 255);
|
||||||
if ($fields['subject'] ?? false) {
|
if ($fields['subject'] ?? false) {
|
||||||
$validator->notEmptyString('subject');
|
$validator->notEmptyString('contact_subject');
|
||||||
} else {
|
} else {
|
||||||
$validator->allowEmptyString('subject');
|
$validator->allowEmptyString('contact_subject');
|
||||||
}
|
}
|
||||||
|
|
||||||
$validator
|
$validator
|
||||||
|
|||||||
@@ -24,9 +24,10 @@
|
|||||||
<?php
|
<?php
|
||||||
echo $this->Form->control('submitted_at');
|
echo $this->Form->control('submitted_at');
|
||||||
echo $this->Form->control('client_ip');
|
echo $this->Form->control('client_ip');
|
||||||
echo $this->Form->control('name');
|
echo $this->Form->control('first_name');
|
||||||
|
echo $this->Form->control('last_name');
|
||||||
echo $this->Form->control('email');
|
echo $this->Form->control('email');
|
||||||
echo $this->Form->control('subject');
|
echo $this->Form->control('contact_subject');
|
||||||
echo $this->Form->control('message');
|
echo $this->Form->control('message');
|
||||||
echo $this->Form->control('confirm_email_sent', ['empty' => true]);
|
echo $this->Form->control('confirm_email_sent', ['empty' => true]);
|
||||||
echo $this->Form->control('backend_email_sent', ['empty' => true]);
|
echo $this->Form->control('backend_email_sent', ['empty' => true]);
|
||||||
|
|||||||
@@ -13,9 +13,9 @@
|
|||||||
<th><?= $this->Paginator->sort('id') ?></th>
|
<th><?= $this->Paginator->sort('id') ?></th>
|
||||||
<th><?= $this->Paginator->sort('submitted_at') ?></th>
|
<th><?= $this->Paginator->sort('submitted_at') ?></th>
|
||||||
<th><?= $this->Paginator->sort('client_ip') ?></th>
|
<th><?= $this->Paginator->sort('client_ip') ?></th>
|
||||||
<th><?= $this->Paginator->sort('name') ?></th>
|
<th><?= $this->Paginator->sort('first_name') . ' ' . $this->Paginator->sort('last_name'); ?></th>
|
||||||
<th><?= $this->Paginator->sort('email') ?></th>
|
<th><?= $this->Paginator->sort('email') ?></th>
|
||||||
<th><?= $this->Paginator->sort('subject') ?></th>
|
<th><?= $this->Paginator->sort('contact_subject') ?></th>
|
||||||
<th><?= $this->Paginator->sort('confirm_email_sent') ?></th>
|
<th><?= $this->Paginator->sort('confirm_email_sent') ?></th>
|
||||||
<th><?= $this->Paginator->sort('backend_email_sent') ?></th>
|
<th><?= $this->Paginator->sort('backend_email_sent') ?></th>
|
||||||
<th class="actions"><?= __('Actions') ?></th>
|
<th class="actions"><?= __('Actions') ?></th>
|
||||||
@@ -27,9 +27,9 @@
|
|||||||
<td><?= h($contactUsFormSubmission->id) ?></td>
|
<td><?= h($contactUsFormSubmission->id) ?></td>
|
||||||
<td><?= h($contactUsFormSubmission->submitted_at) ?></td>
|
<td><?= h($contactUsFormSubmission->submitted_at) ?></td>
|
||||||
<td><?= h($contactUsFormSubmission->client_ip) ?></td>
|
<td><?= h($contactUsFormSubmission->client_ip) ?></td>
|
||||||
<td><?= h($contactUsFormSubmission->name) ?></td>
|
<td><?= h($contactUsFormSubmission->full_name) ?></td>
|
||||||
<td><?= h($contactUsFormSubmission->email) ?></td>
|
<td><?= h($contactUsFormSubmission->email) ?></td>
|
||||||
<td><?= h($contactUsFormSubmission->subject) ?></td>
|
<td><?= h($contactUsFormSubmission->contact_subject) ?></td>
|
||||||
<td><?= h($contactUsFormSubmission->confirm_email_sent) ?></td>
|
<td><?= h($contactUsFormSubmission->confirm_email_sent) ?></td>
|
||||||
<td><?= h($contactUsFormSubmission->backend_email_sent) ?></td>
|
<td><?= h($contactUsFormSubmission->backend_email_sent) ?></td>
|
||||||
<td class="actions">
|
<td class="actions">
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
</aside>
|
</aside>
|
||||||
<div class="column column-80">
|
<div class="column column-80">
|
||||||
<div class="contactUsFormSubmissions view content">
|
<div class="contactUsFormSubmissions view content">
|
||||||
<h3><?= h($contactUsFormSubmission->name) ?></h3>
|
<h3><?= h($contactUsFormSubmission->email) ?></h3>
|
||||||
<table>
|
<table>
|
||||||
<tr>
|
<tr>
|
||||||
<th><?= __('Id') ?></th>
|
<th><?= __('Id') ?></th>
|
||||||
@@ -27,7 +27,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th><?= __('Name') ?></th>
|
<th><?= __('Name') ?></th>
|
||||||
<td><?= h($contactUsFormSubmission->name) ?></td>
|
<td><?= h($contactUsFormSubmission->full_name) ?></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th><?= __('Email') ?></th>
|
<th><?= __('Email') ?></th>
|
||||||
@@ -35,7 +35,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th><?= __('Subject') ?></th>
|
<th><?= __('Subject') ?></th>
|
||||||
<td><?= h($contactUsFormSubmission->subject) ?></td>
|
<td><?= h($contactUsFormSubmission->contact_subject) ?></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th><?= __('Submitted At') ?></th>
|
<th><?= __('Submitted At') ?></th>
|
||||||
|
|||||||
@@ -5,10 +5,6 @@
|
|||||||
*/
|
*/
|
||||||
?>
|
?>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<aside class="column">
|
|
||||||
<div class="side-nav">
|
|
||||||
</div>
|
|
||||||
</aside>
|
|
||||||
<div class="column column-80">
|
<div class="column column-80">
|
||||||
<div class="contactUsFormSubmissions form content">
|
<div class="contactUsFormSubmissions form content">
|
||||||
<?= $this->Form->create($contactUsFormSubmission) ?>
|
<?= $this->Form->create($contactUsFormSubmission) ?>
|
||||||
|
|||||||
@@ -7,10 +7,10 @@
|
|||||||
$fields = \Cake\Core\Configure::readOrFail('ContactUs.fields');
|
$fields = \Cake\Core\Configure::readOrFail('ContactUs.fields');
|
||||||
?>
|
?>
|
||||||
<?php
|
<?php
|
||||||
echo $this->Form->control('name');
|
echo $this->Form->control('first_name');
|
||||||
|
echo $this->Form->control('last_name');
|
||||||
echo isset($fields['email']) && $fields['email'] ? $this->Form->control('email', ['required' => true]) : '';
|
echo isset($fields['email']) && $fields['email'] ? $this->Form->control('email', ['required' => true]) : '';
|
||||||
echo isset($fields['subject']) && $fields['subject'] ? $this->Form->control('subject', ['required' => true]) : '';
|
echo isset($fields['subject']) && $fields['subject'] ? $this->Form->control('subject', ['required' => true]) : '';
|
||||||
echo $this->Form->control('message');
|
echo $this->Form->control('message');
|
||||||
echo isset($fields['captcha']) && $fields['captcha'] ? $this->Captcha->render(['placeholder' => __('Please solve the riddle')]) : '';
|
echo isset($fields['captcha']) && $fields['captcha'] ? $this->Captcha->render(['placeholder' => __('Please solve the riddle')]) : '';
|
||||||
|
|
||||||
?>
|
?>
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ use Cake\Core\Configure;
|
|||||||
<?= __d('cake_contact_us', "A contact us form submission was received at {0}", $contactUsFormSubmission->submitted_at) ?>,
|
<?= __d('cake_contact_us', "A contact us form submission was received at {0}", $contactUsFormSubmission->submitted_at) ?>,
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<?= h($contactUsFormSubmission->name); ?>
|
<?= h($contactUsFormSubmission->full_name); ?>
|
||||||
</p>
|
</p>
|
||||||
<?php if (Configure::read('ContactUs.fields.email', false) && isset($contactUsFormSubmission->email)) : ?>
|
<?php if (Configure::read('ContactUs.fields.email', false) && isset($contactUsFormSubmission->email)) : ?>
|
||||||
<p>
|
<p>
|
||||||
@@ -20,13 +20,13 @@ use Cake\Core\Configure;
|
|||||||
</p>
|
</p>
|
||||||
<?php endif; ?>
|
<?php endif; ?>
|
||||||
|
|
||||||
<?php if (Configure::read('ContactUs.fields.subject', false) && isset($contactUsFormSubmission->subject)) : ?>
|
<?php if (Configure::read('ContactUs.fields.subject', false) && isset($contactUsFormSubmission->contact_subject)) : ?>
|
||||||
<p>
|
<p>
|
||||||
<strong><?= __d('cake_contact_us', 'Subject: ') ?></strong>
|
<strong><?= __d('cake_contact_us', 'Subject: ') ?></strong>
|
||||||
<span><?= h($contactUsFormSubmission->subject); ?></span>
|
<span><?= h($contactUsFormSubmission->contact_subject); ?></span>
|
||||||
</p>
|
</p>
|
||||||
<?php endif; ?>
|
<?php endif; ?>
|
||||||
<p>
|
<p>
|
||||||
<strong><?= __d('cake_contact_us', 'Message: ') ?></strong>
|
<strong><?= __d('cake_contact_us', 'Message: ') ?></strong>
|
||||||
<?= h($contactUsFormSubmission->message); ?>
|
<?= h($contactUsFormSubmission->message); ?>
|
||||||
</p>
|
</p>
|
||||||
|
|||||||
@@ -3,13 +3,17 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace CakeContactUs\Test\Fixture;
|
namespace CakeContactUs\Test\Fixture;
|
||||||
|
|
||||||
|
use Cake\I18n\FrozenTime;
|
||||||
use Cake\TestSuite\Fixture\TestFixture;
|
use Cake\TestSuite\Fixture\TestFixture;
|
||||||
|
use Cake\Utility\Text;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ContactUsFormSubmissionsFixture
|
* ContactUsFormSubmissionsFixture
|
||||||
*/
|
*/
|
||||||
class ContactUsFormSubmissionsFixture extends TestFixture {
|
class ContactUsFormSubmissionsFixture extends TestFixture {
|
||||||
|
|
||||||
|
public string $table = 'contact_us_form_submissions';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Init method
|
* Init method
|
||||||
*
|
*
|
||||||
@@ -18,17 +22,19 @@ class ContactUsFormSubmissionsFixture extends TestFixture {
|
|||||||
public function init(): void {
|
public function init(): void {
|
||||||
$this->records = [
|
$this->records = [
|
||||||
[
|
[
|
||||||
'id' => '76fbe7fb-1949-4670-a3d2-c0b48eb98e8d',
|
'id' => Text::uuid(),
|
||||||
'submitted_at' => '2025-01-03 09:16:50',
|
'submitted_at' => new FrozenTime(),
|
||||||
'client_ip' => 'Lorem ipsum dolor sit amet',
|
'client_ip' => 'cli',
|
||||||
'name' => 'Lorem ipsum dolor sit amet',
|
'first_name' => 'test',
|
||||||
'email' => 'Lorem ipsum dolor sit amet',
|
'last_name' => 'test',
|
||||||
'subject' => 'Lorem ipsum dolor sit amet',
|
'email' => 'test@test.com',
|
||||||
'message' => 'Lorem ipsum dolor sit amet, aliquet feugiat. Convallis morbi fringilla gravida, phasellus feugiat dapibus velit nunc, pulvinar eget sollicitudin venenatis cum nullam, vivamus ut a sed, mollitia lectus. Nulla vestibulum massa neque ut et, id hendrerit sit, feugiat in taciti enim proin nibh, tempor dignissim, rhoncus duis vestibulum nunc mattis convallis.',
|
'contact_subject' => 'subject',
|
||||||
'confirm_email_sent' => '2025-01-03 09:16:50',
|
'message' => 'what are your business hours?',
|
||||||
'backend_email_sent' => '2025-01-03 09:16:50',
|
'confirm_email_sent' => true,
|
||||||
|
'backend_email_sent' => true,
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|
||||||
parent::init();
|
parent::init();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -14,19 +14,20 @@ use CakeContactUs\Model\Entity\ContactUsFormSubmission;
|
|||||||
* CakeContactUs\Controller\Component\ContactUsComponent Test Case
|
* CakeContactUs\Controller\Component\ContactUsComponent Test Case
|
||||||
*/
|
*/
|
||||||
class ContactUsComponentTest extends TestCase {
|
class ContactUsComponentTest extends TestCase {
|
||||||
|
|
||||||
protected ContactUsComponent $component;
|
protected ContactUsComponent $component;
|
||||||
protected Controller $controller;
|
|
||||||
|
protected Controller $controller;
|
||||||
/**
|
|
||||||
|
/**
|
||||||
* Fixtures
|
* Fixtures
|
||||||
*
|
*
|
||||||
* @var array<string>
|
* @var array<string>
|
||||||
*/
|
*/
|
||||||
protected array $fixtures = [
|
protected array $fixtures = [
|
||||||
'plugin.CakeContactUs.ContactUsFormSubmissions',
|
'plugin.CakeContactUs.ContactUsFormSubmissions',
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* setUp method
|
* setUp method
|
||||||
*
|
*
|
||||||
@@ -34,11 +35,11 @@ class ContactUsComponentTest extends TestCase {
|
|||||||
*/
|
*/
|
||||||
protected function setUp(): void {
|
protected function setUp(): void {
|
||||||
parent::setUp();
|
parent::setUp();
|
||||||
$request = new ServerRequest();
|
$request = new ServerRequest();
|
||||||
$this->controller = new Controller($request);
|
$this->controller = new Controller($request);
|
||||||
$registry = new ComponentRegistry($this->controller);
|
$registry = new ComponentRegistry($this->controller);
|
||||||
|
|
||||||
$this->component = new ContactUsComponent($registry);
|
$this->component = new ContactUsComponent($registry);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -51,27 +52,35 @@ class ContactUsComponentTest extends TestCase {
|
|||||||
|
|
||||||
parent::tearDown();
|
parent::tearDown();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testNewContactUsFormWithoutCaptcha()
|
|
||||||
{
|
|
||||||
$this->assertInstanceOf(ContactUsFormSubmission::class, $this->component->newContactUsForm());
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testNewContactUsFormWithCaptcha()
|
|
||||||
{
|
|
||||||
$this->component->setConfig('requireCaptcha', true);
|
|
||||||
$this->assertInstanceOf(ContactUsFormSubmission::class, $this->component->newContactUsForm());
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testProcessContactUsFormSaved()
|
/**
|
||||||
{
|
* @return void
|
||||||
$numSubmissionsBefore = $this->fetchTable('CakeContactUs/ContactUsFormSubmissions')->find()->count();
|
*/
|
||||||
$result = $this->component->processContactUsForm($this->component->newContactUsForm(), [
|
public function testNewContactUsFormWithoutCaptcha() {
|
||||||
'name' => 'Jane Doe',
|
$this->assertInstanceOf(ContactUsFormSubmission::class, $this->component->newContactUsForm());
|
||||||
'email' => 'test@example.com',
|
}
|
||||||
]);
|
|
||||||
$this->assertNotInstanceOf(ContactUsFormSubmission::class, $result);
|
/**
|
||||||
$numSubmissionsAfter = $this->fetchTable('CakeContactUs/ContactUsFormSubmissions')->find()->count();
|
* @return void
|
||||||
$this->assertEquals($numSubmissionsBefore + 1, $numSubmissionsAfter);
|
*/
|
||||||
}
|
public function testNewContactUsFormWithCaptcha() {
|
||||||
|
$this->component->setConfig('requireCaptcha', true);
|
||||||
|
$this->assertInstanceOf(ContactUsFormSubmission::class, $this->component->newContactUsForm());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testProcessContactUsFormSaved() {
|
||||||
|
$numSubmissionsBefore = $this->fetchTable('CakeContactUs.ContactUsFormSubmissions')->find()->count();
|
||||||
|
$result = $this->component->processContactUsForm($this->component->newContactUsForm(), [
|
||||||
|
'first_name' => 'Jane Doe',
|
||||||
|
'email' => 'test@example.com',
|
||||||
|
'message' => 'contact us message',
|
||||||
|
]);
|
||||||
|
$this->assertNotInstanceOf(ContactUsFormSubmission::class, $result);
|
||||||
|
$numSubmissionsAfter = $this->fetchTable('CakeContactUs.ContactUsFormSubmissions')->find()->count();
|
||||||
|
$this->assertEquals($numSubmissionsBefore + 1, $numSubmissionsAfter);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -105,7 +105,7 @@ class ContactUsFormSubmissionsControllerTest extends TestCase {
|
|||||||
'action' => 'add',
|
'action' => 'add',
|
||||||
];
|
];
|
||||||
$data = [
|
$data = [
|
||||||
'name' => 'valid name',
|
'first_name' => 'valid name',
|
||||||
'email' => 'valid_email@test.com',
|
'email' => 'valid_email@test.com',
|
||||||
'message' => 'valid message goes here',
|
'message' => 'valid message goes here',
|
||||||
];
|
];
|
||||||
@@ -156,7 +156,7 @@ class ContactUsFormSubmissionsControllerTest extends TestCase {
|
|||||||
'action' => 'add',
|
'action' => 'add',
|
||||||
];
|
];
|
||||||
$data = [
|
$data = [
|
||||||
'name' => 'valid name',
|
'first_name' => 'valid name',
|
||||||
'email' => 'valid_email@test.com',
|
'email' => 'valid_email@test.com',
|
||||||
'message' => 'valid message goes here',
|
'message' => 'valid message goes here',
|
||||||
];
|
];
|
||||||
@@ -203,7 +203,7 @@ class ContactUsFormSubmissionsControllerTest extends TestCase {
|
|||||||
'action' => 'add',
|
'action' => 'add',
|
||||||
];
|
];
|
||||||
$data = [
|
$data = [
|
||||||
'name' => 'valid name',
|
'first_name' => 'valid name',
|
||||||
'email' => 'valid_email@test.com',
|
'email' => 'valid_email@test.com',
|
||||||
'message' => 'valid message goes here',
|
'message' => 'valid message goes here',
|
||||||
];
|
];
|
||||||
@@ -251,7 +251,7 @@ class ContactUsFormSubmissionsControllerTest extends TestCase {
|
|||||||
'action' => 'add',
|
'action' => 'add',
|
||||||
];
|
];
|
||||||
$data = [
|
$data = [
|
||||||
'name' => 'valid name',
|
'first_name' => 'valid name',
|
||||||
'email' => 'valid_email@test.com',
|
'email' => 'valid_email@test.com',
|
||||||
'message' => 'valid message goes here',
|
'message' => 'valid message goes here',
|
||||||
];
|
];
|
||||||
@@ -290,7 +290,7 @@ class ContactUsFormSubmissionsControllerTest extends TestCase {
|
|||||||
'action' => 'add',
|
'action' => 'add',
|
||||||
];
|
];
|
||||||
$data = [
|
$data = [
|
||||||
'name' => 'valid name',
|
'first_name' => 'valid name',
|
||||||
'email' => 'not_valid_email',
|
'email' => 'not_valid_email',
|
||||||
'message' => 'this is a valid message ',
|
'message' => 'this is a valid message ',
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -141,6 +141,7 @@ ConnectionManager::setConfig('test', [
|
|||||||
'timezone' => 'UTC',
|
'timezone' => 'UTC',
|
||||||
'quoteIdentifiers' => false,
|
'quoteIdentifiers' => false,
|
||||||
'cacheMetadata' => true,
|
'cacheMetadata' => true,
|
||||||
|
'log' => true,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -156,16 +157,20 @@ ConnectionManager::setConfig('test', [
|
|||||||
// Load a schema dump file.
|
// Load a schema dump file.
|
||||||
//(new SchemaLoader())->loadSqlFiles('tests/schema.sql', 'test');
|
//(new SchemaLoader())->loadSqlFiles('tests/schema.sql', 'test');
|
||||||
if (!defined('SECOND')) {
|
if (!defined('SECOND')) {
|
||||||
define('SECOND', 1);
|
define('SECOND', 1);
|
||||||
define('MINUTE', 60);
|
define('MINUTE', 60);
|
||||||
define('HOUR', 3600);
|
define('HOUR', 3600);
|
||||||
define('DAY', 86400);
|
define('DAY', 86400);
|
||||||
define('WEEK', 604800);
|
define('WEEK', 604800);
|
||||||
define('MONTH', 2592000);
|
define('MONTH', 2592000);
|
||||||
define('YEAR', 31536000);
|
define('YEAR', 31536000);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
$migrator = new Migrator();
|
$migrator = new Migrator();
|
||||||
$migrator->run(['plugin' => 'CakeContactUs']);
|
$migrator->runMany([
|
||||||
$migrator->run(['plugin' => 'Captcha']);
|
// Run app migrations on test connection.
|
||||||
|
['connection' => 'test'],
|
||||||
|
// Run plugin migrations on test connection.
|
||||||
|
['plugin' => 'CakeContactUs'],
|
||||||
|
['plugin' => 'Captcha'],
|
||||||
|
]);
|
||||||
|
|||||||
Reference in New Issue
Block a user