Compare commits
92 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 5b1ca4241c | |||
| c8d6263c0f | |||
| bd1972d894 | |||
| fa29477c80 | |||
| 20ee958db9 | |||
| 157616582f | |||
| 0f7bf0ab44 | |||
| 10dc56c7cb | |||
| d19842007c | |||
| c5bd16de74 | |||
| e580e5430c | |||
| da9717265c | |||
| 6c6b33cacc | |||
| 5509d1c92e | |||
| 16d789f061 | |||
| 3450b6263b | |||
| bcc1e833c1 | |||
| fea4c315f3 | |||
| 3ac57e05b0 | |||
| b0230e7050 | |||
| 70f78a5f9c | |||
| 814ec50d3d | |||
| e60ce3f0dc | |||
| 6140bd8671 | |||
| e318a4fb9a | |||
| 9818410889 | |||
| 62e1e3b97c | |||
| 1bb4d5f426 | |||
| a65485e56a | |||
| 5e6ba85639 | |||
| 3cf891a7d2 | |||
| 6e19f30ff5 | |||
| 95079885c5 | |||
| 14dab54f2c | |||
| 180fba596a | |||
| 69e2503a36 | |||
| c06ae55919 | |||
| 6098d1091a | |||
| 2a8038835f | |||
| 984dec37e2 | |||
| d91fbd5a46 | |||
| 668954c1c4 | |||
| adbae19fca | |||
| 39de680b66 | |||
| 2e81a4da82 | |||
| 61782be11c | |||
| e7fe4655d1 | |||
| 561be4bce6 | |||
| 51fef30364 | |||
| 89a4b4cf3e | |||
| 3eb1229590 | |||
| f174e9ec34 | |||
| 928369cff7 | |||
| 29f4800e1a | |||
| e706a67427 | |||
| 672de694ee | |||
| c69eac3258 | |||
| eca62e53c9 | |||
| 62cd3badfe | |||
| 4bd2568d94 | |||
| b3c7947c67 | |||
| 6710f44646 | |||
| f89be6bd19 | |||
| 9696336abc | |||
| fec1c5ad57 | |||
| a6e5e86c9e | |||
| 1357b98883 | |||
| fa18a6f422 | |||
| ada130329f | |||
| 4c489adf9b | |||
| c8a2ad4d48 | |||
| d54a6cbdf5 | |||
| e8f4c82f59 | |||
| 24eb2c2bee | |||
| 94edb7f556 | |||
| 3f5b3f02ad | |||
| a270922b4b | |||
| 8c607ae610 | |||
| d3d4cf512f | |||
| efae352c35 | |||
| a1614bace2 | |||
| 62ce863f2c | |||
| b6ef3b29f6 | |||
| 1f178628a2 | |||
| 67b725cb65 | |||
| 648d2c3eb5 | |||
| 428c10b547 | |||
| dd705be420 | |||
| a1ba1257ac | |||
| 9569145054 | |||
| 13602bdd73 | |||
| 1ef91305d4 |
33
.github/workflows-disabled/ci.yml
vendored
Normal file
33
.github/workflows-disabled/ci.yml
vendored
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
name: CI
|
||||||
|
run-name: ${{ github.actor}} runs CI
|
||||||
|
|
||||||
|
on: [push]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
phpstan:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
# - uses: php-actions/composer@v6
|
||||||
|
# env:
|
||||||
|
# COMPOSER_ROOT_VERSION: dev-master
|
||||||
|
- name: "Restore result cache"
|
||||||
|
uses: actions/cache/restore@v4
|
||||||
|
with:
|
||||||
|
path: ./tmp
|
||||||
|
key: "result-cache-v1-${{ matrix.php-version }}-${{ github.run_id }}"
|
||||||
|
restore-keys: |
|
||||||
|
result-cache-v1-${{ matrix.php-version }}-
|
||||||
|
- name: PHPStan Static Analysis
|
||||||
|
uses: php-actions/phpstan@v3
|
||||||
|
with:
|
||||||
|
path: src/
|
||||||
|
- name: "Save result cache"
|
||||||
|
uses: actions/cache/save@v4
|
||||||
|
if: always()
|
||||||
|
with:
|
||||||
|
path: ./tmp
|
||||||
|
key: "result-cache-v1-${{ matrix.php-version }}-${{ github.run_id }}"
|
||||||
|
# - name: PHPunit Tests
|
||||||
|
# run: |
|
||||||
|
# vendor/bin/phpunit
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,2 +1,3 @@
|
|||||||
vendor
|
vendor/
|
||||||
composer.lock
|
composer.lock
|
||||||
|
tools/
|
||||||
|
|||||||
@@ -54,7 +54,8 @@ return [
|
|||||||
// Note that the **only** effect of choosing `'5.6'` is to infer that functions removed in php 7.0 exist.
|
// Note that the **only** effect of choosing `'5.6'` is to infer that functions removed in php 7.0 exist.
|
||||||
// (See `backward_compatibility_checks` for additional options)
|
// (See `backward_compatibility_checks` for additional options)
|
||||||
// Automatically inferred from composer.json requirement for "php" of ">=8.2"
|
// Automatically inferred from composer.json requirement for "php" of ">=8.2"
|
||||||
'target_php_version' => '8.1',
|
'target_php_version' => '8.2',
|
||||||
|
"minimum_target_php_version" => "8.2",
|
||||||
|
|
||||||
// If enabled, missing properties will be created when
|
// If enabled, missing properties will be created when
|
||||||
// they are first seen. If false, we'll report an
|
// they are first seen. If false, we'll report an
|
||||||
@@ -370,6 +371,7 @@ return [
|
|||||||
'file_list' => [
|
'file_list' => [
|
||||||
"./test/configs/config.php",
|
"./test/configs/config.php",
|
||||||
"./test/configs/config.other.php",
|
"./test/configs/config.other.php",
|
||||||
|
"./test/configs/config.path.php",
|
||||||
"./test/configs/config.master.php",
|
"./test/configs/config.master.php",
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<phive xmlns="https://phar.io/phive">
|
<phive xmlns="https://phar.io/phive">
|
||||||
<phar name="phpunit" version="^9.6" installed="9.6.13" location="./tools/phpunit" copy="false"/>
|
<phar name="phpunit" version="^9.6" installed="9.6.19" location="./tools/phpunit" copy="false"/>
|
||||||
<phar name="phpcs" version="^3.7.2" installed="3.7.2" location="./tools/phpcs" copy="false"/>
|
<phar name="phpcs" version="^3.7.2" installed="3.10.0" location="./tools/phpcs" copy="false"/>
|
||||||
<phar name="phpcbf" version="^3.7.2" installed="3.7.2" location="./tools/phpcbf" copy="false"/>
|
<phar name="phpcbf" version="^3.7.2" installed="3.10.0" location="./tools/phpcbf" copy="false"/>
|
||||||
<phar name="psalm" version="^5.15.0" installed="5.15.0" location="./tools/psalm" copy="false"/>
|
<phar name="psalm" version="^5.15.0" installed="5.24.0" location="./tools/psalm" copy="false"/>
|
||||||
<phar name="phpstan" version="^1.10.37" installed="1.10.37" location="./tools/phpstan" copy="false"/>
|
<phar name="phpstan" version="^1.10.37" installed="1.11.1" location="./tools/phpstan" copy="false"/>
|
||||||
<phar name="phan" version="^5.4.2" installed="5.4.2" location="./tools/phan" copy="false"/>
|
<phar name="phan" version="^5.4.2" installed="5.4.3" location="./tools/phan" copy="false"/>
|
||||||
</phive>
|
</phive>
|
||||||
|
|||||||
2
.shellcheckrc
Normal file
2
.shellcheckrc
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
shell=bash
|
||||||
|
external-sources=true
|
||||||
39
ReadMe.md
39
ReadMe.md
@@ -9,20 +9,45 @@ For local install only
|
|||||||
- Template\SmartyExtended
|
- Template\SmartyExtended
|
||||||
- Admin\EditBase
|
- Admin\EditBase
|
||||||
|
|
||||||
## Setup from central composer
|
## Publish to gitea or gitlab server
|
||||||
|
|
||||||
Setup from gitea internal servers
|
Currently there are only gitea and gitlab supported, github does not have support for composer packages
|
||||||
|
|
||||||
```sh
|
`publish\publish.sh go` will run the publish script
|
||||||
composer config repositories.git.egplusww.jp.Composer composer https://git.egplusww.jp/api/packages/Composer/composer
|
|
||||||
|
All the configuration is done in the `publish\.env.deploy` file
|
||||||
|
|
||||||
|
```ini
|
||||||
|
# downlaod file name is "Repository name" "-" "version" where
|
||||||
|
# version is "vN.N.N"
|
||||||
|
GITEA_PUBLISH=1
|
||||||
|
GITEA_UPLOAD_FILENAME="Upload-File-Name";
|
||||||
|
GITEA_USER=gitea-user
|
||||||
|
GITEA_TOKEN=gitea-tokek
|
||||||
|
GITEA_URL_DL=https://[gitea.hostname]/[to/package/folder]/archive
|
||||||
|
GITEA_URL_PUSH=https://[gitea.hostname]/api/packages/[organization]/composer
|
||||||
|
|
||||||
|
GITLAB_PUBLISH=1
|
||||||
|
GITLAB_URL=gitlab URl to repository
|
||||||
|
GITLAB_DEPLOY_TOKEN=gitlab-token
|
||||||
```
|
```
|
||||||
|
|
||||||
Alternative setup composer local zip file repot:
|
At the moment there is only one gitea or gitlab target setable
|
||||||
`composer config repositories.composer.egplusww.jp composer http://composer.egplusww.jp`
|
|
||||||
|
## Setup from central composer
|
||||||
|
|
||||||
|
Setup from gitea servers
|
||||||
|
|
||||||
|
[hostname] is the hostname for your gitea server (or wherever this is published)
|
||||||
|
[OrgName] is the organization name where the composer packages are hosted
|
||||||
|
|
||||||
|
```sh
|
||||||
|
composer config repositories.[hostname].Composer composer https://[hostname]/api/packages/[OrgName]/composer
|
||||||
|
```
|
||||||
|
|
||||||
## Install package
|
## Install package
|
||||||
|
|
||||||
`composer require egrajp/corelibs-composer-all:^8.0`
|
`composer require egrajp/corelibs-composer-all:^9.0`
|
||||||
|
|
||||||
## Tests
|
## Tests
|
||||||
|
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
"description": "CoreLibs in a composer package",
|
"description": "CoreLibs in a composer package",
|
||||||
"type": "library",
|
"type": "library",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
"keywords": ["corelib", "logging", "database", "templating", "tools"],
|
||||||
"autoload": {
|
"autoload": {
|
||||||
"psr-4": {
|
"psr-4": {
|
||||||
"CoreLibs\\": "src/"
|
"CoreLibs\\": "src/"
|
||||||
@@ -20,8 +21,13 @@
|
|||||||
"psr/log": "^3.0@dev"
|
"psr/log": "^3.0@dev"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
|
"phpstan/phpstan": "^2.0",
|
||||||
|
"phpstan/phpdoc-parser": "^2.0",
|
||||||
|
"phpstan/phpstan-deprecation-rules": "^2.0",
|
||||||
|
"phan/phan": "^5.4",
|
||||||
"egrajp/smarty-extended": "^4.3",
|
"egrajp/smarty-extended": "^4.3",
|
||||||
"gullevek/dotenv": "dev-master"
|
"gullevek/dotenv": "dev-master",
|
||||||
|
"phpunit/phpunit": "^9"
|
||||||
},
|
},
|
||||||
"repositories": {
|
"repositories": {
|
||||||
"git.egplusww.jp.Composer": {
|
"git.egplusww.jp.Composer": {
|
||||||
|
|||||||
18
phpcs.xml
Normal file
18
phpcs.xml
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
<?xml version="1.0"?>
|
||||||
|
<ruleset name="MyStandard">
|
||||||
|
<description>PSR12 override rules (strict, standard). Switch spaces indent to tab.</description>
|
||||||
|
<arg name="tab-width" value="4"/>
|
||||||
|
<rule ref="PSR1"/>
|
||||||
|
<rule ref="PSR12">
|
||||||
|
<!-- turn off white space check for tab -->
|
||||||
|
<exclude name="Generic.WhiteSpace.DisallowTabIndent"/>
|
||||||
|
</rule>
|
||||||
|
<!-- no space indent, must be tab, 4 is tab iwdth -->
|
||||||
|
<rule ref="Generic.WhiteSpace.DisallowSpaceIndent"/>
|
||||||
|
<rule ref="Generic.WhiteSpace.ScopeIndent">
|
||||||
|
<properties>
|
||||||
|
<property name="indent" value="4"/>
|
||||||
|
<property name="tabIndent" value="true"/>
|
||||||
|
</properties>
|
||||||
|
</rule>
|
||||||
|
</ruleset>
|
||||||
@@ -8,5 +8,6 @@ $_SERVER['HTTP_HOST'] = 'soba.tokyo.tequila.jp';
|
|||||||
// for whatever reason it does not load that from the confing.master.php
|
// for whatever reason it does not load that from the confing.master.php
|
||||||
// for includes/admin_header.php
|
// for includes/admin_header.php
|
||||||
define('BASE_NAME', '');
|
define('BASE_NAME', '');
|
||||||
|
define('CONTENT_PATH', '');
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
includes:
|
includes:
|
||||||
- phpstan-conditional.php
|
- phpstan-conditional.php
|
||||||
parameters:
|
parameters:
|
||||||
tmpDir: /tmp/phpstan-corelibs-composer
|
tmpDir: %currentWorkingDirectory%/tmp/phpstan-corelibs-composer
|
||||||
level: 8 # max is now 9
|
level: 8 # max is now 9
|
||||||
checkMissingCallableSignature: true
|
checkMissingCallableSignature: true
|
||||||
treatPhpDocTypesAsCertain: false
|
treatPhpDocTypesAsCertain: false
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
9.8.2
|
9.23.1
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
BASE_FOLDER=$(dirname $(readlink -f $0))"/";
|
BASE_FOLDER=$(dirname "$(readlink -f "$0")")"/";
|
||||||
PACKAGE_DOWNLOAD="${BASE_FOLDER}package-download/";
|
PACKAGE_DOWNLOAD="${BASE_FOLDER}package-download/";
|
||||||
if [ ! -d "${PACKAGE_DOWNLOAD}" ]; then
|
if [ ! -d "${PACKAGE_DOWNLOAD}" ]; then
|
||||||
mkdir "${PACKAGE_DOWNLOAD}";
|
mkdir "${PACKAGE_DOWNLOAD}";
|
||||||
@@ -15,30 +15,35 @@ if [ -z "${VERSION}" ]; then
|
|||||||
fi;
|
fi;
|
||||||
# compare version, if different or newer, deploy
|
# compare version, if different or newer, deploy
|
||||||
if [ -f "${file_last_published}" ]; then
|
if [ -f "${file_last_published}" ]; then
|
||||||
LAST_PUBLISHED_VERSION=$(cat ${file_last_published});
|
LAST_PUBLISHED_VERSION=$(cat "${file_last_published}");
|
||||||
if $(dpkg --compare-versions "${VERSION}" le "${LAST_PUBLISHED_VERSION}"); then
|
if dpkg --compare-versions "${VERSION}" le "${LAST_PUBLISHED_VERSION}"; then
|
||||||
echo "git tag version ${VERSION} is not newer than previous published version ${LAST_PUBLISHED_VERSION}";
|
echo "git tag version ${VERSION} is not newer than previous published version ${LAST_PUBLISHED_VERSION}";
|
||||||
exit;
|
exit;
|
||||||
fi;
|
fi;
|
||||||
fi;
|
fi;
|
||||||
|
|
||||||
# read in the .env.deploy file and we must have
|
# read in the .env.deploy file and we must have
|
||||||
|
# for gitea
|
||||||
|
# GITEA_PUBLISH: must be set with a value to trigger publish run
|
||||||
# GITEA_UPLOAD_FILENAME
|
# GITEA_UPLOAD_FILENAME
|
||||||
# GITLAB_USER
|
|
||||||
# GITLAB_TOKEN
|
|
||||||
# GITLAB_URL
|
|
||||||
# GITEA_USER
|
# GITEA_USER
|
||||||
# GITEA_DEPLOY_TOKEN
|
# GITEA_DEPLOY_TOKEN
|
||||||
# GITEA_URL_DL
|
# GITEA_URL_DL
|
||||||
# GITEA_URL_PUSH
|
# GITEA_URL_PUSH
|
||||||
|
# for gitlab
|
||||||
|
# GITLAB_PUBLISH: must be set with a value to trigger publish run
|
||||||
|
# GITLAB_USER
|
||||||
|
# GITLAB_TOKEN
|
||||||
|
# GITLAB_URL
|
||||||
if [ ! -f "${BASE_FOLDER}.env.deploy" ]; then
|
if [ ! -f "${BASE_FOLDER}.env.deploy" ]; then
|
||||||
echo "Deploy enviroment file .env.deploy is missing";
|
echo "Deploy enviroment file .env.deploy is missing";
|
||||||
exit;
|
exit;
|
||||||
fi;
|
fi;
|
||||||
set -o allexport;
|
set -o allexport;
|
||||||
cd ${BASE_FOLDER};
|
cd "${BASE_FOLDER}" || exit;
|
||||||
|
# shellcheck source=.env.deploy
|
||||||
source .env.deploy;
|
source .env.deploy;
|
||||||
cd -;
|
cd - || exit;
|
||||||
set +o allexport;
|
set +o allexport;
|
||||||
|
|
||||||
if [ "${go_flag}" != "go" ]; then
|
if [ "${go_flag}" != "go" ]; then
|
||||||
@@ -50,31 +55,42 @@ fi;
|
|||||||
|
|
||||||
echo "[START]";
|
echo "[START]";
|
||||||
# gitea
|
# gitea
|
||||||
if [ ! -z "${GITEA_UPLOAD_FILENAME}" ] &&
|
# skip iof
|
||||||
[ ! -z "${GITEA_URL_DL}" ] && [ ! -z "${GITEA_URL_PUSH}" ] &&
|
if [ -n "${GITEA_PUBLISH}" ]; then
|
||||||
[ ! -z "${GITEA_USER}" ] && [ ! -z "${GITEA_TOKEN}" ]; then
|
if [ -n "${GITEA_UPLOAD_FILENAME}" ] &&
|
||||||
curl -LJO \
|
[ -n "${GITEA_URL_DL}" ] && [ -n "${GITEA_URL_PUSH}" ] &&
|
||||||
--output-dir "${PACKAGE_DOWNLOAD}" \
|
[ -n "${GITEA_USER}" ] && [ -n "${GITEA_TOKEN}" ]; then
|
||||||
${GITEA_URL_DL}/v${VERSION}.zip;
|
if [ ! -f "${PACKAGE_DOWNLOAD}${GITEA_UPLOAD_FILENAME}-v${VERSION}.zip" ]; then
|
||||||
curl --user ${GITEA_USER}:${GITEA_TOKEN} \
|
curl -LJO \
|
||||||
--upload-file "${PACKAGE_DOWNLOAD}${GITEA_UPLOAD_FILENAME}-v${VERSION}.zip" \
|
--output-dir "${PACKAGE_DOWNLOAD}" \
|
||||||
${GITEA_URL_PUSH}?version=${VERSION};
|
"${GITEA_URL_DL}"/v"${VERSION}".zip;
|
||||||
echo "${VERSION}" > "${file_last_published}";
|
fi;
|
||||||
else
|
if [ ! -f "${PACKAGE_DOWNLOAD}${GITEA_UPLOAD_FILENAME}-v${VERSION}.zip" ]; then
|
||||||
echo "Missing either GITEA_UPLOAD_FILENAME, GITEA_URL_DL, GITEA_URL_PUSH, GITEA_USER or GITEA_TOKEN environment variable";
|
echo "Version file does not exist for ${VERSION}";
|
||||||
|
else
|
||||||
|
curl --user "${GITEA_USER}":"${GITEA_TOKEN}" \
|
||||||
|
--upload-file "${PACKAGE_DOWNLOAD}${GITEA_UPLOAD_FILENAME}-v${VERSION}.zip" \
|
||||||
|
"${GITEA_URL_PUSH}"?version="${VERSION}";
|
||||||
|
echo "${VERSION}" > "${file_last_published}";
|
||||||
|
fi;
|
||||||
|
else
|
||||||
|
echo "Missing either GITEA_UPLOAD_FILENAME, GITEA_URL_DL, GITEA_URL_PUSH, GITEA_USER or GITEA_TOKEN environment variable";
|
||||||
|
fi;
|
||||||
fi;
|
fi;
|
||||||
|
|
||||||
# gitlab
|
# gitlab
|
||||||
if [ ! -z "${GITLAB_URL}" ] && [ ! -z "${GITLAB_DEPLOY_TOKEN}" ]; then
|
if [ -n "${GITLAB_PUBLISH}" ]; then
|
||||||
curl --data tag=v${VERSION} \
|
if [ -n "${GITLAB_URL}" ] && [ -n "${GITLAB_DEPLOY_TOKEN}" ]; then
|
||||||
--header "Deploy-Token: ${GITLAB_DEPLOY_TOKEN}" \
|
curl --data tag=v"${VERSION}" \
|
||||||
"${GITLAB_URL}";
|
--header "Deploy-Token: ${GITLAB_DEPLOY_TOKEN}" \
|
||||||
curl --data branch=master \
|
"${GITLAB_URL}";
|
||||||
--header "Deploy-Token: ${GITLAB_DEPLOY_TOKEN}" \
|
curl --data branch=master \
|
||||||
"${GITLAB_URL}";
|
--header "Deploy-Token: ${GITLAB_DEPLOY_TOKEN}" \
|
||||||
echo "${VERSION}" > "${file_last_published}";
|
"${GITLAB_URL}";
|
||||||
else
|
echo "${VERSION}" > "${file_last_published}";
|
||||||
echo "Missing GITLAB_DEPLOY_TOKEN environment variable";
|
else
|
||||||
|
echo "Missing GITLAB_URL or GITLAB_DEPLOY_TOKEN environment variable";
|
||||||
|
fi;
|
||||||
fi;
|
fi;
|
||||||
echo "";
|
echo "";
|
||||||
echo "[DONE]";
|
echo "[DONE]";
|
||||||
|
|||||||
@@ -69,12 +69,17 @@ declare(strict_types=1);
|
|||||||
namespace CoreLibs\ACL;
|
namespace CoreLibs\ACL;
|
||||||
|
|
||||||
use CoreLibs\Security\Password;
|
use CoreLibs\Security\Password;
|
||||||
|
use CoreLibs\Create\Uids;
|
||||||
use CoreLibs\Convert\Json;
|
use CoreLibs\Convert\Json;
|
||||||
|
|
||||||
class Login
|
class Login
|
||||||
{
|
{
|
||||||
/** @var ?int the user id var*/
|
/** @var ?int the user id var*/
|
||||||
private ?int $euid;
|
private ?int $euid;
|
||||||
|
/** @var ?string the user cuid (note will be super seeded with uuid v4 later) */
|
||||||
|
private ?string $ecuid;
|
||||||
|
/** @var ?string UUIDv4, will superseed the ecuid and replace euid as login id */
|
||||||
|
private ?string $ecuuid;
|
||||||
/** @var string _GET/_POST loginUserId parameter for non password login */
|
/** @var string _GET/_POST loginUserId parameter for non password login */
|
||||||
private string $login_user_id = '';
|
private string $login_user_id = '';
|
||||||
/** @var string source, either _GET or _POST or empty */
|
/** @var string source, either _GET or _POST or empty */
|
||||||
@@ -193,6 +198,12 @@ class Login
|
|||||||
/** @var bool */
|
/** @var bool */
|
||||||
private bool $login_is_ajax_page = false;
|
private bool $login_is_ajax_page = false;
|
||||||
|
|
||||||
|
// logging
|
||||||
|
/** @var array<string> list of allowed types for edit log write */
|
||||||
|
private const WRITE_TYPES = ['BINARY', 'BZIP2', 'LZIP', 'STRING', 'SERIAL', 'JSON'];
|
||||||
|
/** @var array<string> list of available write types for log */
|
||||||
|
private array $write_types_available = [];
|
||||||
|
|
||||||
// settings
|
// settings
|
||||||
/** @var array<string,mixed> options */
|
/** @var array<string,mixed> options */
|
||||||
private array $options = [];
|
private array $options = [];
|
||||||
@@ -361,9 +372,6 @@ class Login
|
|||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|
||||||
// init default ACL list array
|
|
||||||
$_SESSION['DEFAULT_ACL_LIST'] = [];
|
|
||||||
$_SESSION['DEFAULT_ACL_LIST_TYPE'] = [];
|
|
||||||
// read the current edit_access_right list into an array
|
// read the current edit_access_right list into an array
|
||||||
$q = "SELECT level, type, name FROM edit_access_right "
|
$q = "SELECT level, type, name FROM edit_access_right "
|
||||||
. "WHERE level >= 0 ORDER BY level";
|
. "WHERE level >= 0 ORDER BY level";
|
||||||
@@ -376,8 +384,12 @@ class Login
|
|||||||
$this->default_acl_list_type[(string)$res['type']] = (int)$res['level'];
|
$this->default_acl_list_type[(string)$res['type']] = (int)$res['level'];
|
||||||
}
|
}
|
||||||
// write that into the session
|
// write that into the session
|
||||||
$_SESSION['DEFAULT_ACL_LIST'] = $this->default_acl_list;
|
$this->session->setMany([
|
||||||
$_SESSION['DEFAULT_ACL_LIST_TYPE'] = $this->default_acl_list_type;
|
'DEFAULT_ACL_LIST' => $this->default_acl_list,
|
||||||
|
'DEFAULT_ACL_LIST_TYPE' => $this->default_acl_list_type,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$this->loginSetEditLogWriteTypeAvailable();
|
||||||
|
|
||||||
// this will be deprecated
|
// this will be deprecated
|
||||||
if ($this->options['auto_login'] === true) {
|
if ($this->options['auto_login'] === true) {
|
||||||
@@ -567,7 +579,7 @@ class Login
|
|||||||
// set path
|
// set path
|
||||||
$options['locale_path'] = BASE . INCLUDES . LOCALE;
|
$options['locale_path'] = BASE . INCLUDES . LOCALE;
|
||||||
}
|
}
|
||||||
$_SESSION['LOCALE_PATH'] = $options['locale_path'];
|
$this->session->set('LOCALE_PATH', $options['locale_path']);
|
||||||
// LANG: LOCALE
|
// LANG: LOCALE
|
||||||
if (empty($options['site_locale'])) {
|
if (empty($options['site_locale'])) {
|
||||||
trigger_error(
|
trigger_error(
|
||||||
@@ -602,7 +614,7 @@ class Login
|
|||||||
$options['set_domain'] = str_replace(DIRECTORY_SEPARATOR, '', CONTENT_PATH);
|
$options['set_domain'] = str_replace(DIRECTORY_SEPARATOR, '', CONTENT_PATH);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$_SESSION['DEFAULT_DOMAIN'] = $options['site_domain'];
|
$this->session->set('DEFAULT_DOMAIN', $options['site_domain']);
|
||||||
// LANG: ENCODING
|
// LANG: ENCODING
|
||||||
if (empty($options['site_encoding'])) {
|
if (empty($options['site_encoding'])) {
|
||||||
trigger_error(
|
trigger_error(
|
||||||
@@ -757,7 +769,7 @@ class Login
|
|||||||
}
|
}
|
||||||
// have to get the global stuff here for setting it later
|
// have to get the global stuff here for setting it later
|
||||||
// we have to get the themes in here too
|
// we have to get the themes in here too
|
||||||
$q = "SELECT eu.edit_user_id, eu.username, eu.password, "
|
$q = "SELECT eu.edit_user_id, eu.cuid, eu.cuuid, eu.username, eu.password, "
|
||||||
. "eu.edit_group_id, "
|
. "eu.edit_group_id, "
|
||||||
. "eg.name AS edit_group_name, eu.admin, "
|
. "eg.name AS edit_group_name, eu.admin, "
|
||||||
// additinal acl lists
|
// additinal acl lists
|
||||||
@@ -888,7 +900,14 @@ class Login
|
|||||||
}
|
}
|
||||||
// normal user processing
|
// normal user processing
|
||||||
// set class var and session var
|
// set class var and session var
|
||||||
$_SESSION['EUID'] = $this->euid = (int)$res['edit_user_id'];
|
$this->euid = (int)$res['edit_user_id'];
|
||||||
|
$this->ecuid = (string)$res['cuid'];
|
||||||
|
$this->ecuuid = (string)$res['cuuid'];
|
||||||
|
$this->session->setMany([
|
||||||
|
'EUID' => $this->euid,
|
||||||
|
'ECUID' => $this->ecuid,
|
||||||
|
'ECUUID' => $this->ecuuid,
|
||||||
|
]);
|
||||||
// check if user is okay
|
// check if user is okay
|
||||||
$this->loginCheckPermissions();
|
$this->loginCheckPermissions();
|
||||||
if ($this->login_error == 0) {
|
if ($this->login_error == 0) {
|
||||||
@@ -901,27 +920,39 @@ class Login
|
|||||||
. "WHERE edit_user_id = " . $this->euid;
|
. "WHERE edit_user_id = " . $this->euid;
|
||||||
$this->db->dbExec($q);
|
$this->db->dbExec($q);
|
||||||
}
|
}
|
||||||
// now set all session vars and read page permissions
|
$locale = $res['locale'] ?? 'en';
|
||||||
$_SESSION['DEBUG_ALL'] = $this->db->dbBoolean($res['debug']);
|
$encoding = $res['encoding'] ?? 'UTF-8';
|
||||||
$_SESSION['DB_DEBUG'] = $this->db->dbBoolean($res['db_debug']);
|
$this->session->setMany([
|
||||||
// general info for user logged in
|
// now set all session vars and read page permissions
|
||||||
$_SESSION['USER_NAME'] = $res['username'];
|
'DEBUG_ALL' => $this->db->dbBoolean($res['debug']),
|
||||||
$_SESSION['ADMIN'] = $res['admin'];
|
'DB_DEBUG' => $this->db->dbBoolean($res['db_debug']),
|
||||||
$_SESSION['GROUP_NAME'] = $res['edit_group_name'];
|
// general info for user logged in
|
||||||
$_SESSION['USER_ACL_LEVEL'] = $res['user_level'];
|
'USER_NAME' => $res['username'],
|
||||||
$_SESSION['USER_ACL_TYPE'] = $res['user_type'];
|
'ADMIN' => $res['admin'],
|
||||||
$_SESSION['USER_ADDITIONAL_ACL'] = Json::jsonConvertToArray($res['user_additional_acl']);
|
'GROUP_NAME' => $res['edit_group_name'],
|
||||||
$_SESSION['GROUP_ACL_LEVEL'] = $res['group_level'];
|
'USER_ACL_LEVEL' => $res['user_level'],
|
||||||
$_SESSION['GROUP_ACL_TYPE'] = $res['group_type'];
|
'USER_ACL_TYPE' => $res['user_type'],
|
||||||
$_SESSION['GROUP_ADDITIONAL_ACL'] = Json::jsonConvertToArray($res['group_additional_acl']);
|
'USER_ADDITIONAL_ACL' => Json::jsonConvertToArray($res['user_additional_acl']),
|
||||||
// deprecated TEMPLATE setting
|
'GROUP_ACL_LEVEL' => $res['group_level'],
|
||||||
$_SESSION['TEMPLATE'] = $res['template'] ? $res['template'] : '';
|
'GROUP_ACL_TYPE' => $res['group_type'],
|
||||||
$_SESSION['HEADER_COLOR'] = !empty($res['second_header_color']) ?
|
'GROUP_ADDITIONAL_ACL' => Json::jsonConvertToArray($res['group_additional_acl']),
|
||||||
$res['second_header_color'] :
|
// deprecated TEMPLATE setting
|
||||||
$res['first_header_color'];
|
'TEMPLATE' => $res['template'] ? $res['template'] : '',
|
||||||
|
'HEADER_COLOR' => !empty($res['second_header_color']) ?
|
||||||
|
$res['second_header_color'] :
|
||||||
|
$res['first_header_color'],
|
||||||
|
// LANGUAGE/LOCALE/ENCODING:
|
||||||
|
'LANG' => $locale,
|
||||||
|
'DEFAULT_CHARSET' => $encoding,
|
||||||
|
'DEFAULT_LOCALE' => $locale . '.' . strtoupper($encoding),
|
||||||
|
'DEFAULT_LANG' => $locale . '_' . strtolower(str_replace('-', '', $encoding))
|
||||||
|
]);
|
||||||
// missing # before, this is for legacy data, will be deprecated
|
// missing # before, this is for legacy data, will be deprecated
|
||||||
if (preg_match("/^[\dA-Fa-f]{6,8}$/", $_SESSION['HEADER_COLOR'])) {
|
if (
|
||||||
$_SESSION['HEADER_COLOR'] = '#' . $_SESSION['HEADER_COLOR'];
|
!empty($this->session->get('HEADER_COLOR')) &&
|
||||||
|
preg_match("/^[\dA-Fa-f]{6,8}$/", $this->session->get('HEADER_COLOR'))
|
||||||
|
) {
|
||||||
|
$this->session->set('HEADER_COLOR', '#' . $this->session->get('HEADER_COLOR'));
|
||||||
}
|
}
|
||||||
// TODO: make sure that header color is valid:
|
// TODO: make sure that header color is valid:
|
||||||
// # + 6 hex
|
// # + 6 hex
|
||||||
@@ -930,13 +961,6 @@ class Login
|
|||||||
// rgb: nnn.n for each
|
// rgb: nnn.n for each
|
||||||
// hsl: nnn.n for first, nnn.n% for 2nd, 3rd
|
// hsl: nnn.n for first, nnn.n% for 2nd, 3rd
|
||||||
// Check\Colors::validateColor()
|
// Check\Colors::validateColor()
|
||||||
// LANGUAGE/LOCALE/ENCODING:
|
|
||||||
$_SESSION['LANG'] = $res['locale'] ?? 'en';
|
|
||||||
$_SESSION['DEFAULT_CHARSET'] = $res['encoding'] ?? 'UTF-8';
|
|
||||||
$_SESSION['DEFAULT_LOCALE'] = $_SESSION['LANG']
|
|
||||||
. '.' . strtoupper($_SESSION['DEFAULT_CHARSET']);
|
|
||||||
$_SESSION['DEFAULT_LANG'] = $_SESSION['LANG'] . '_'
|
|
||||||
. strtolower(str_replace('-', '', $_SESSION['DEFAULT_CHARSET']));
|
|
||||||
// reset any login error count for this user
|
// reset any login error count for this user
|
||||||
if ($res['login_error_count'] > 0) {
|
if ($res['login_error_count'] > 0) {
|
||||||
$q = "UPDATE edit_user "
|
$q = "UPDATE edit_user "
|
||||||
@@ -960,10 +984,7 @@ class Login
|
|||||||
. "AND ear.edit_access_right_id = epa.edit_access_right_id "
|
. "AND ear.edit_access_right_id = epa.edit_access_right_id "
|
||||||
. "AND epa.enabled = 1 AND epa.edit_group_id = " . $res["edit_group_id"] . " "
|
. "AND epa.enabled = 1 AND epa.edit_group_id = " . $res["edit_group_id"] . " "
|
||||||
. "ORDER BY ep.order_number";
|
. "ORDER BY ep.order_number";
|
||||||
while ($res = $this->db->dbReturn($q)) {
|
while (is_array($res = $this->db->dbReturn($q))) {
|
||||||
if (!is_array($res)) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
// page id array for sub data readout
|
// page id array for sub data readout
|
||||||
$edit_page_ids[$res['edit_page_id']] = $res['cuid'];
|
$edit_page_ids[$res['edit_page_id']] = $res['cuid'];
|
||||||
// create the array for pages
|
// create the array for pages
|
||||||
@@ -1029,8 +1050,10 @@ class Login
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
// write back the pages data to the output array
|
// write back the pages data to the output array
|
||||||
$_SESSION['PAGES'] = $pages;
|
$this->session->setMany([
|
||||||
$_SESSION['PAGES_ACL_LEVEL'] = $pages_acl;
|
'PAGES' => $pages,
|
||||||
|
'PAGES_ACL_LEVEL' => $pages_acl,
|
||||||
|
]);
|
||||||
// load the edit_access user rights
|
// load the edit_access user rights
|
||||||
$q = "SELECT ea.edit_access_id, level, type, ea.name, "
|
$q = "SELECT ea.edit_access_id, level, type, ea.name, "
|
||||||
. "ea.color, ea.uid, edit_default, ea.additional_acl "
|
. "ea.color, ea.uid, edit_default, ea.additional_acl "
|
||||||
@@ -1042,6 +1065,7 @@ class Login
|
|||||||
$unit_access = [];
|
$unit_access = [];
|
||||||
$eauid = [];
|
$eauid = [];
|
||||||
$unit_acl = [];
|
$unit_acl = [];
|
||||||
|
$unit_uid = [];
|
||||||
while (is_array($res = $this->db->dbReturn($q))) {
|
while (is_array($res = $this->db->dbReturn($q))) {
|
||||||
// read edit access data fields and drop them into the unit access array
|
// read edit access data fields and drop them into the unit access array
|
||||||
$q_sub = "SELECT name, value "
|
$q_sub = "SELECT name, value "
|
||||||
@@ -1065,16 +1089,19 @@ class Login
|
|||||||
];
|
];
|
||||||
// set the default unit
|
// set the default unit
|
||||||
if ($res['edit_default']) {
|
if ($res['edit_default']) {
|
||||||
$_SESSION['UNIT_DEFAULT'] = (int)$res['edit_access_id'];
|
$this->session->set('UNIT_DEFAULT', (int)$res['edit_access_id']);
|
||||||
}
|
}
|
||||||
$_SESSION['UNIT_UID'][$res['uid']] = (int)$res['edit_access_id'];
|
$unit_uid[$res['uid']] = (int)$res['edit_access_id'];
|
||||||
// sub arrays for simple access
|
// sub arrays for simple access
|
||||||
array_push($eauid, $res['edit_access_id']);
|
array_push($eauid, $res['edit_access_id']);
|
||||||
$unit_acl[$res['edit_access_id']] = $res['level'];
|
$unit_acl[$res['edit_access_id']] = $res['level'];
|
||||||
}
|
}
|
||||||
$_SESSION['UNIT'] = $unit_access;
|
$this->session->setMany([
|
||||||
$_SESSION['UNIT_ACL_LEVEL'] = $unit_acl;
|
'UNIT_UID' => $unit_uid,
|
||||||
$_SESSION['EAID'] = $eauid;
|
'UNIT' => $unit_access,
|
||||||
|
'UNIT_ACL_LEVEL' => $unit_acl,
|
||||||
|
'EAID' => $eauid,
|
||||||
|
]);
|
||||||
} // user has permission to THIS page
|
} // user has permission to THIS page
|
||||||
} // user was not enabled or other login error
|
} // user was not enabled or other login error
|
||||||
if ($this->login_error && is_array($res)) {
|
if ($this->login_error && is_array($res)) {
|
||||||
@@ -1135,6 +1162,9 @@ class Login
|
|||||||
// username (login), group name
|
// username (login), group name
|
||||||
$this->acl['user_name'] = $_SESSION['USER_NAME'];
|
$this->acl['user_name'] = $_SESSION['USER_NAME'];
|
||||||
$this->acl['group_name'] = $_SESSION['GROUP_NAME'];
|
$this->acl['group_name'] = $_SESSION['GROUP_NAME'];
|
||||||
|
// edit user cuid
|
||||||
|
$this->acl['ecuid'] = $_SESSION['ECUID'];
|
||||||
|
$this->acl['ecuuid'] = $_SESSION['ECUUID'];
|
||||||
// set additional acl
|
// set additional acl
|
||||||
$this->acl['additional_acl'] = [
|
$this->acl['additional_acl'] = [
|
||||||
'user' => $_SESSION['USER_ADDITIONAL_ACL'],
|
'user' => $_SESSION['USER_ADDITIONAL_ACL'],
|
||||||
@@ -1167,7 +1197,7 @@ class Login
|
|||||||
$this->acl['base'] = (int)$_SESSION['USER_ACL_LEVEL'];
|
$this->acl['base'] = (int)$_SESSION['USER_ACL_LEVEL'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$_SESSION['BASE_ACL_LEVEL'] = $this->acl['base'];
|
$this->session->set('BASE_ACL_LEVEL', $this->acl['base']);
|
||||||
|
|
||||||
// set the current page acl
|
// set the current page acl
|
||||||
// start with base acl
|
// start with base acl
|
||||||
@@ -1303,11 +1333,9 @@ class Login
|
|||||||
{
|
{
|
||||||
$is_valid_password = true;
|
$is_valid_password = true;
|
||||||
// check for valid in regex arrays in list
|
// check for valid in regex arrays in list
|
||||||
if (is_array($this->password_valid_chars)) {
|
foreach ($this->password_valid_chars as $password_valid_chars) {
|
||||||
foreach ($this->password_valid_chars as $password_valid_chars) {
|
if (!preg_match("/$password_valid_chars/", $password)) {
|
||||||
if (!preg_match("/$password_valid_chars/", $password)) {
|
$is_valid_password = false;
|
||||||
$is_valid_password = false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// check for min length
|
// check for min length
|
||||||
@@ -1430,7 +1458,7 @@ class Login
|
|||||||
$data = 'Illegal user for password change: ' . $this->pw_username;
|
$data = 'Illegal user for password change: ' . $this->pw_username;
|
||||||
}
|
}
|
||||||
// log this password change attempt
|
// log this password change attempt
|
||||||
$this->writeLog($event, $data, $this->login_error, $this->pw_username);
|
$this->writeEditLog($event, $data, $this->login_error, $this->pw_username);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1571,7 +1599,7 @@ class Login
|
|||||||
$username = $res['username'];
|
$username = $res['username'];
|
||||||
}
|
}
|
||||||
} // if euid is set, get username (or try)
|
} // if euid is set, get username (or try)
|
||||||
$this->writeLog($event, '', $this->login_error, $username);
|
$this->writeEditLog($event, '', $this->login_error, $username);
|
||||||
} // write log under certain settings
|
} // write log under certain settings
|
||||||
// now close DB connection
|
// now close DB connection
|
||||||
// $this->error_msg = $this->_login();
|
// $this->error_msg = $this->_login();
|
||||||
@@ -1727,6 +1755,8 @@ HTML;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: LOGGING
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* writes detailed data into the edit user log table (keep log what user does)
|
* writes detailed data into the edit user log table (keep log what user does)
|
||||||
*
|
*
|
||||||
@@ -1736,7 +1766,7 @@ HTML;
|
|||||||
* @param string $username login user username
|
* @param string $username login user username
|
||||||
* @return void has no return
|
* @return void has no return
|
||||||
*/
|
*/
|
||||||
private function writeLog(
|
private function writeEditLog(
|
||||||
string $event,
|
string $event,
|
||||||
string $data,
|
string $data,
|
||||||
string|int $error = '',
|
string|int $error = '',
|
||||||
@@ -1754,50 +1784,191 @@ HTML;
|
|||||||
'_GET' => $_GET,
|
'_GET' => $_GET,
|
||||||
'_POST' => $_POST,
|
'_POST' => $_POST,
|
||||||
'_FILES' => $_FILES,
|
'_FILES' => $_FILES,
|
||||||
'error' => $this->login_error
|
'error' => $this->login_error,
|
||||||
|
'data' => $data,
|
||||||
];
|
];
|
||||||
$data_binary = $this->db->dbEscapeBytea((string)bzcompress(serialize($_data_binary)));
|
$_action_set = [
|
||||||
// SQL querie for log entry
|
'action' => $this->action,
|
||||||
$q = "INSERT INTO edit_log "
|
'action_id' => $this->username,
|
||||||
. "(username, password, euid, event_date, event, error, data, data_binary, page, "
|
'action_flag' => (string)$this->login_error,
|
||||||
. "ip, user_agent, referer, script_name, query_string, server_name, http_host, "
|
'action_value' => (string)$this->permission_okay,
|
||||||
. "http_accept, http_accept_charset, http_accept_encoding, session_id, "
|
];
|
||||||
. "action, action_id, action_yes, action_flag, action_menu, action_loaded, "
|
|
||||||
. "action_value, action_error) "
|
$this->writeLog($event, $_data_binary, $_action_set, $error, $username);
|
||||||
. "VALUES ('" . $this->db->dbEscapeString($username) . "', 'PASSWORD', "
|
}
|
||||||
. ($this->euid ? $this->euid : 'NULL') . ", "
|
|
||||||
. "NOW(), '" . $this->db->dbEscapeString($event) . "', "
|
/**
|
||||||
. "'" . $this->db->dbEscapeString((string)$error) . "', "
|
* writes all action vars plus other info into edit_log table
|
||||||
. "'" . $this->db->dbEscapeString($data) . "', '" . $data_binary . "', "
|
* this is for public class
|
||||||
. "'" . $this->page_name . "', ";
|
*
|
||||||
foreach (
|
* phpcs:disable Generic.Files.LineLength
|
||||||
[
|
* @param string $event [default=''] any kind of event description,
|
||||||
'REMOTE_ADDR', 'HTTP_USER_AGENT', 'HTTP_REFERER', 'SCRIPT_FILENAME',
|
* @param string|array<mixed> $data [default=''] any kind of data related to that event
|
||||||
'QUERY_STRING', 'SERVER_NAME', 'HTTP_HOST', 'HTTP_ACCEPT',
|
* @param array{action?:?string,action_id?:null|string|int,action_sub_id?:null|string|int,action_yes?:null|string|int|bool,action_flag?:?string,action_menu?:?string,action_loaded?:?string,action_value?:?string,action_type?:?string,action_error?:?string} $action_set [default=[]] action set names
|
||||||
'HTTP_ACCEPT_CHARSET', 'HTTP_ACCEPT_ENCODING'
|
* @param string|int $error error id (mostly an int)
|
||||||
] as $server_code
|
* @param string $write_type [default=JSON] write type can be
|
||||||
) {
|
* JSON, STRING/SERIEAL, BINARY/BZIP or ZLIB
|
||||||
if (array_key_exists($server_code, $_SERVER)) {
|
* @param string|null $db_schema [default=null] override target schema
|
||||||
$q .= "'" . $this->db->dbEscapeString($_SERVER[$server_code]) . "', ";
|
* @return void
|
||||||
} else {
|
* phpcs:enable Generic.Files.LineLength
|
||||||
$q .= "NULL, ";
|
*/
|
||||||
}
|
public function writeLog(
|
||||||
|
string $event = '',
|
||||||
|
string|array $data = '',
|
||||||
|
array $action_set = [],
|
||||||
|
string|int $error = '',
|
||||||
|
string $username = '',
|
||||||
|
string $write_type = 'JSON',
|
||||||
|
?string $db_schema = null
|
||||||
|
): void {
|
||||||
|
$data_binary = '';
|
||||||
|
$data_write = '';
|
||||||
|
|
||||||
|
// check if write type is valid, if not fallback to JSON
|
||||||
|
if (!in_array(strtoupper($write_type), $this->write_types_available)) {
|
||||||
|
$this->log->warning('Write type not in allowed array, fallback to JSON', context:[
|
||||||
|
"write_type" => $write_type,
|
||||||
|
"write_list" => $this->write_types_available,
|
||||||
|
]);
|
||||||
|
$write_type = 'JSON';
|
||||||
|
}
|
||||||
|
switch ($write_type) {
|
||||||
|
case 'BINARY':
|
||||||
|
case 'BZIP':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea((string)bzcompress(serialize($data)));
|
||||||
|
$data_write = Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'BZIP',
|
||||||
|
'message' => 'see bzip compressed data_binary field'
|
||||||
|
]);
|
||||||
|
break;
|
||||||
|
case 'ZLIB':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea((string)gzcompress(serialize($data)));
|
||||||
|
$data_write = Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'ZLIB',
|
||||||
|
'message' => 'see zlib compressed data_binary field'
|
||||||
|
]);
|
||||||
|
break;
|
||||||
|
case 'STRING':
|
||||||
|
case 'SERIAL':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea(Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'SERIAL',
|
||||||
|
'message' => 'see serial string data field'
|
||||||
|
]));
|
||||||
|
$data_write = serialize($data);
|
||||||
|
break;
|
||||||
|
case 'JSON':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea(Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'JSON',
|
||||||
|
'message' => 'see json string data field'
|
||||||
|
]));
|
||||||
|
// must be converted to array
|
||||||
|
if (!is_array($data)) {
|
||||||
|
$data = ["data" => $data];
|
||||||
|
}
|
||||||
|
$data_write = Json::jsonConvertArrayTo($data);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$this->log->alert('Invalid type for data compression was set', context:[
|
||||||
|
"write_type" => $write_type
|
||||||
|
]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var string $DB_SCHEMA check schema */
|
||||||
|
$DB_SCHEMA = 'public';
|
||||||
|
if ($db_schema !== null) {
|
||||||
|
$DB_SCHEMA = $db_schema;
|
||||||
|
} elseif (!empty($this->db->dbGetSchema())) {
|
||||||
|
$DB_SCHEMA = $this->db->dbGetSchema();
|
||||||
|
}
|
||||||
|
$q = <<<SQL
|
||||||
|
INSERT INTO {DB_SCHEMA}.edit_log (
|
||||||
|
username, euid, ecuid, ecuuid, event_date, event, error, data, data_binary, page,
|
||||||
|
ip, user_agent, referer, script_name, query_string, server_name, http_host,
|
||||||
|
http_accept, http_accept_charset, http_accept_encoding, session_id,
|
||||||
|
action, action_id, action_sub_id, action_yes, action_flag, action_menu, action_loaded,
|
||||||
|
action_value, action_type, action_error
|
||||||
|
) VALUES (
|
||||||
|
$1, $2, $3, $4, NOW(), $5, $6, $7, $8, $9,
|
||||||
|
$10, $11, $12, $13, $14, $15, $16,
|
||||||
|
$17, $18, $19, $20,
|
||||||
|
$21, $22, $23, $24, $25, $26, $27,
|
||||||
|
$28, $29, $30
|
||||||
|
)
|
||||||
|
SQL;
|
||||||
|
$this->db->dbExecParams(
|
||||||
|
str_replace(
|
||||||
|
['{DB_SCHEMA}'],
|
||||||
|
[$DB_SCHEMA],
|
||||||
|
$q
|
||||||
|
),
|
||||||
|
[
|
||||||
|
// row 1
|
||||||
|
empty($username) ? $this->session->get('USER_NAME') ?? '' : $username,
|
||||||
|
is_numeric($this->session->get('EUID')) ?
|
||||||
|
$this->session->get('EUID') : null,
|
||||||
|
is_string($this->session->get('ECUID')) ?
|
||||||
|
$this->session->get('ECUID') : null,
|
||||||
|
!empty($this->session->get('ECUUID')) && Uids::validateUuuidv4($this->session->get('ECUUID')) ?
|
||||||
|
$this->session->get('ECUUID') : null,
|
||||||
|
(string)$event,
|
||||||
|
(string)$error,
|
||||||
|
$data_write,
|
||||||
|
$data_binary,
|
||||||
|
(string)$this->page_name,
|
||||||
|
// row 2
|
||||||
|
$_SERVER["REMOTE_ADDR"] ?? null,
|
||||||
|
$_SERVER['HTTP_USER_AGENT'] ?? null,
|
||||||
|
$_SERVER['HTTP_REFERER'] ?? null,
|
||||||
|
$_SERVER['SCRIPT_FILENAME'] ?? null,
|
||||||
|
$_SERVER['QUERY_STRING'] ?? null,
|
||||||
|
$_SERVER['SERVER_NAME'] ?? null,
|
||||||
|
$_SERVER['HTTP_HOST'] ?? null,
|
||||||
|
// row 3
|
||||||
|
$_SERVER['HTTP_ACCEPT'] ?? null,
|
||||||
|
$_SERVER['HTTP_ACCEPT_CHARSET'] ?? null,
|
||||||
|
$_SERVER['HTTP_ACCEPT_ENCODING'] ?? null,
|
||||||
|
$this->session->getSessionId() !== '' ?
|
||||||
|
$this->session->getSessionId() : null,
|
||||||
|
// row 4
|
||||||
|
$action_set['action'] ?? null,
|
||||||
|
$action_set['action_id'] ?? null,
|
||||||
|
$action_set['action_sub_id'] ?? null,
|
||||||
|
$action_set['action_yes'] ?? null,
|
||||||
|
$action_set['action_flag'] ?? null,
|
||||||
|
$action_set['action_menu'] ?? null,
|
||||||
|
$action_set['action_loaded'] ?? null,
|
||||||
|
$action_set['action_value'] ?? null,
|
||||||
|
$action_set['action_type'] ?? null,
|
||||||
|
$action_set['action_error'] ?? null,
|
||||||
|
],
|
||||||
|
'NULL'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the write types that are allowed
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function loginSetEditLogWriteTypeAvailable()
|
||||||
|
{
|
||||||
|
// check what edit log data write types are allowed
|
||||||
|
$this->write_types_available = self::WRITE_TYPES;
|
||||||
|
if (!function_exists('bzcompress')) {
|
||||||
|
$this->write_types_available = array_diff($this->write_types_available, ['BINARY', 'BZIP']);
|
||||||
|
}
|
||||||
|
if (!function_exists('gzcompress')) {
|
||||||
|
$this->write_types_available = array_diff($this->write_types_available, ['LZIP']);
|
||||||
}
|
}
|
||||||
$q .= "'" . $this->session->getSessionId() . "', ";
|
|
||||||
$q .= "'" . $this->db->dbEscapeString($this->action) . "', ";
|
|
||||||
$q .= "'" . $this->db->dbEscapeString($this->username) . "', ";
|
|
||||||
$q .= "NULL, ";
|
|
||||||
$q .= "'" . $this->db->dbEscapeString((string)$this->login_error) . "', ";
|
|
||||||
$q .= "NULL, NULL, ";
|
|
||||||
$q .= "'" . $this->db->dbEscapeString((string)$this->permission_okay) . "', ";
|
|
||||||
$q .= "NULL)";
|
|
||||||
$this->db->dbExec($q, 'NULL');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// *************************************************************************
|
// *************************************************************************
|
||||||
// **** PUBLIC INTERNAL
|
// **** PUBLIC INTERNAL
|
||||||
// *************************************************************************
|
// *************************************************************************
|
||||||
|
|
||||||
|
// MARK: LOGIN CALL
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Main call that needs to be run to actaully check for login
|
* Main call that needs to be run to actaully check for login
|
||||||
* If this is not called, no login checks are done, unless the class
|
* If this is not called, no login checks are done, unless the class
|
||||||
@@ -1866,7 +2037,10 @@ HTML;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// if there is none, there is none, saves me POST/GET check
|
// if there is none, there is none, saves me POST/GET check
|
||||||
$this->euid = array_key_exists('EUID', $_SESSION) ? (int)$_SESSION['EUID'] : 0;
|
$this->euid = (int)($this->session->get('EUID') ?? 0);
|
||||||
|
// TODO: allow load from cuid
|
||||||
|
// $this->ecuid = (string)($this->session->get('ECUID') ?? '');
|
||||||
|
// $this->ecuuid = (string)($this->session->get('ECUUID') ?? '');
|
||||||
// get login vars, are so, can't be changed
|
// get login vars, are so, can't be changed
|
||||||
// prepare
|
// prepare
|
||||||
// pass on vars to Object vars
|
// pass on vars to Object vars
|
||||||
@@ -1947,6 +2121,8 @@ HTML;
|
|||||||
$this->loginSetAcl();
|
$this->loginSetAcl();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: setters/getters
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns current set login_html content
|
* Returns current set login_html content
|
||||||
*
|
*
|
||||||
@@ -2116,6 +2292,8 @@ HTML;
|
|||||||
$this->session->sessionDestroy();
|
$this->session->sessionDestroy();
|
||||||
// unset euid
|
// unset euid
|
||||||
$this->euid = null;
|
$this->euid = null;
|
||||||
|
$this->ecuid = null;
|
||||||
|
$this->ecuuid = null;
|
||||||
// then prints the login screen again
|
// then prints the login screen again
|
||||||
$this->permission_okay = false;
|
$this->permission_okay = false;
|
||||||
}
|
}
|
||||||
@@ -2133,11 +2311,12 @@ HTML;
|
|||||||
if (empty($this->euid)) {
|
if (empty($this->euid)) {
|
||||||
return $this->permission_okay;
|
return $this->permission_okay;
|
||||||
}
|
}
|
||||||
|
// euid must match ecuid and ecuuid
|
||||||
// bail for previous wrong page match, eg if method is called twice
|
// bail for previous wrong page match, eg if method is called twice
|
||||||
if ($this->login_error == 103) {
|
if ($this->login_error == 103) {
|
||||||
return $this->permission_okay;
|
return $this->permission_okay;
|
||||||
}
|
}
|
||||||
$q = "SELECT ep.filename, "
|
$q = "SELECT ep.filename, eu.cuid, eu.cuuid, "
|
||||||
// base lock flags
|
// base lock flags
|
||||||
. "eu.deleted, eu.enabled, eu.locked, "
|
. "eu.deleted, eu.enabled, eu.locked, "
|
||||||
// date based lock
|
// date based lock
|
||||||
@@ -2203,6 +2382,13 @@ HTML;
|
|||||||
} else {
|
} else {
|
||||||
$this->login_error = 103;
|
$this->login_error = 103;
|
||||||
}
|
}
|
||||||
|
// set ECUID
|
||||||
|
$this->ecuid = (string)$res['cuid'];
|
||||||
|
$this->ecuuid = (string)$res['cuuid'];
|
||||||
|
$this->session->setMany([
|
||||||
|
'ECUID' => $this->ecuid,
|
||||||
|
'ECUUID' => $this->ecuuid,
|
||||||
|
]);
|
||||||
// if called from public, so we can check if the permissions are ok
|
// if called from public, so we can check if the permissions are ok
|
||||||
return $this->permission_okay;
|
return $this->permission_okay;
|
||||||
}
|
}
|
||||||
@@ -2348,13 +2534,12 @@ HTML;
|
|||||||
{
|
{
|
||||||
if (
|
if (
|
||||||
$edit_access_id !== null &&
|
$edit_access_id !== null &&
|
||||||
isset($_SESSION['UNIT']) &&
|
is_array($this->session->get('UNIT')) &&
|
||||||
is_array($_SESSION['UNIT']) &&
|
!array_key_exists($edit_access_id, $this->session->get('UNIT'))
|
||||||
!array_key_exists($edit_access_id, $_SESSION['UNIT'])
|
|
||||||
) {
|
) {
|
||||||
$edit_access_id = null;
|
$edit_access_id = null;
|
||||||
if (is_numeric($_SESSION['UNIT_DEFAULT'])) {
|
if (is_numeric($this->session->get('UNIT_DEFAULT'))) {
|
||||||
$edit_access_id = (int)$_SESSION['UNIT_DEFAULT'];
|
$edit_access_id = (int)$this->session->get('UNIT_DEFAULT');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return $edit_access_id;
|
return $edit_access_id;
|
||||||
@@ -2485,7 +2670,7 @@ HTML;
|
|||||||
*/
|
*/
|
||||||
public function loginGetHeaderColor(): ?string
|
public function loginGetHeaderColor(): ?string
|
||||||
{
|
{
|
||||||
return $_SESSION['HEADER_COLOR'] ?? null;
|
return $this->session->get('HEADER_COLOR');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -2496,7 +2681,7 @@ HTML;
|
|||||||
public function loginGetPages(): array
|
public function loginGetPages(): array
|
||||||
{
|
{
|
||||||
|
|
||||||
return $_SESSION['PAGES'] ?? [];
|
return $this->session->get('PAGES');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -2508,6 +2693,26 @@ HTML;
|
|||||||
{
|
{
|
||||||
return (string)$this->euid;
|
return (string)$this->euid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the current set ECUID (edit user cuid)
|
||||||
|
*
|
||||||
|
* @return string ECUID as string
|
||||||
|
*/
|
||||||
|
public function loginGetEcuid(): string
|
||||||
|
{
|
||||||
|
return (string)$this->ecuid;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the current set ECUUID (edit user cuuid)
|
||||||
|
*
|
||||||
|
* @return string ECUUID as string
|
||||||
|
*/
|
||||||
|
public function loginGetEcuuid(): string
|
||||||
|
{
|
||||||
|
return (string)$this->ecuuid;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -31,6 +31,9 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace CoreLibs\Admin;
|
namespace CoreLibs\Admin;
|
||||||
|
|
||||||
|
use CoreLibs\Create\Uids;
|
||||||
|
use CoreLibs\Convert\Json;
|
||||||
|
|
||||||
class Backend
|
class Backend
|
||||||
{
|
{
|
||||||
// page name
|
// page name
|
||||||
@@ -42,7 +45,7 @@ class Backend
|
|||||||
/** @var array<string> */
|
/** @var array<string> */
|
||||||
public array $action_list = [
|
public array $action_list = [
|
||||||
'action', 'action_id', 'action_sub_id', 'action_yes', 'action_flag',
|
'action', 'action_id', 'action_sub_id', 'action_yes', 'action_flag',
|
||||||
'action_menu', 'action_value', 'action_error', 'action_loaded'
|
'action_menu', 'action_value', 'action_type', 'action_error', 'action_loaded'
|
||||||
];
|
];
|
||||||
/** @var string */
|
/** @var string */
|
||||||
public string $action;
|
public string $action;
|
||||||
@@ -61,20 +64,31 @@ class Backend
|
|||||||
/** @var string */
|
/** @var string */
|
||||||
public string $action_value;
|
public string $action_value;
|
||||||
/** @var string */
|
/** @var string */
|
||||||
|
public string $action_type;
|
||||||
|
/** @var string */
|
||||||
public string $action_error;
|
public string $action_error;
|
||||||
|
|
||||||
// ACL array variable if we want to set acl data from outisde
|
// ACL array variable if we want to set acl data from outisde
|
||||||
/** @var array<mixed> */
|
/** @var array<mixed> */
|
||||||
public array $acl = [];
|
public array $acl = [];
|
||||||
/** @var int */
|
/** @var int */
|
||||||
public int $default_acl;
|
public int $default_acl;
|
||||||
|
|
||||||
// queue key
|
// queue key
|
||||||
/** @var string */
|
/** @var string */
|
||||||
public string $queue_key;
|
public string $queue_key;
|
||||||
|
|
||||||
|
/** @var array<string> list of allowed types for edit log write */
|
||||||
|
private const WRITE_TYPES = ['BINARY', 'BZIP2', 'LZIP', 'STRING', 'SERIAL', 'JSON'];
|
||||||
|
/** @var array<string> list of available write types for log */
|
||||||
|
private array $write_types_available = [];
|
||||||
|
|
||||||
// the current active edit access id
|
// the current active edit access id
|
||||||
/** @var int|null */
|
/** @var int|null */
|
||||||
public int|null $edit_access_id;
|
public int|null $edit_access_id;
|
||||||
/** @var string */
|
/** @var string */
|
||||||
public string $page_name;
|
public string $page_name;
|
||||||
|
|
||||||
// error/warning/info messages
|
// error/warning/info messages
|
||||||
/** @var array<mixed> */
|
/** @var array<mixed> */
|
||||||
public array $messages = [];
|
public array $messages = [];
|
||||||
@@ -84,6 +98,7 @@ class Backend
|
|||||||
public bool $warning = false;
|
public bool $warning = false;
|
||||||
/** @var bool */
|
/** @var bool */
|
||||||
public bool $info = false;
|
public bool $info = false;
|
||||||
|
|
||||||
// internal lang & encoding vars
|
// internal lang & encoding vars
|
||||||
/** @var string */
|
/** @var string */
|
||||||
public string $lang_dir = '';
|
public string $lang_dir = '';
|
||||||
@@ -95,6 +110,7 @@ class Backend
|
|||||||
public string $domain;
|
public string $domain;
|
||||||
/** @var string */
|
/** @var string */
|
||||||
public string $encoding;
|
public string $encoding;
|
||||||
|
|
||||||
/** @var \CoreLibs\Logging\Logging logger */
|
/** @var \CoreLibs\Logging\Logging logger */
|
||||||
public \CoreLibs\Logging\Logging $log;
|
public \CoreLibs\Logging\Logging $log;
|
||||||
/** @var \CoreLibs\DB\IO database */
|
/** @var \CoreLibs\DB\IO database */
|
||||||
@@ -103,6 +119,7 @@ class Backend
|
|||||||
public \CoreLibs\Language\L10n $l;
|
public \CoreLibs\Language\L10n $l;
|
||||||
/** @var \CoreLibs\Create\Session session class */
|
/** @var \CoreLibs\Create\Session session class */
|
||||||
public \CoreLibs\Create\Session $session;
|
public \CoreLibs\Create\Session $session;
|
||||||
|
|
||||||
// smarty publics [end processing in smarty class]
|
// smarty publics [end processing in smarty class]
|
||||||
/** @var array<mixed> */
|
/** @var array<mixed> */
|
||||||
public array $DATA = [];
|
public array $DATA = [];
|
||||||
@@ -117,18 +134,20 @@ class Backend
|
|||||||
/**
|
/**
|
||||||
* main class constructor
|
* main class constructor
|
||||||
*
|
*
|
||||||
* @param \CoreLibs\DB\IO $db Database connection class
|
* @param \CoreLibs\DB\IO $db Database connection class
|
||||||
* @param \CoreLibs\Logging\Logging $log Logging class
|
* @param \CoreLibs\Logging\Logging $log Logging class
|
||||||
* @param \CoreLibs\Create\Session $session Session interface class
|
* @param \CoreLibs\Create\Session $session Session interface class
|
||||||
* @param \CoreLibs\Language\L10n $l10n l10n language class
|
* @param \CoreLibs\Language\L10n $l10n l10n language class
|
||||||
* @param int|null $set_default_acl_level Default ACL level
|
* @param int|null $set_default_acl_level [default=null] Default ACL level
|
||||||
|
* @param bool $init_action_vars [default=true] If the action vars should be set
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(
|
||||||
\CoreLibs\DB\IO $db,
|
\CoreLibs\DB\IO $db,
|
||||||
\CoreLibs\Logging\Logging $log,
|
\CoreLibs\Logging\Logging $log,
|
||||||
\CoreLibs\Create\Session $session,
|
\CoreLibs\Create\Session $session,
|
||||||
\CoreLibs\Language\L10n $l10n,
|
\CoreLibs\Language\L10n $l10n,
|
||||||
?int $set_default_acl_level = null
|
?int $set_default_acl_level = null,
|
||||||
|
bool $init_action_vars = true
|
||||||
) {
|
) {
|
||||||
// attach db class
|
// attach db class
|
||||||
$this->db = $db;
|
$this->db = $db;
|
||||||
@@ -151,9 +170,9 @@ class Backend
|
|||||||
// set the page name
|
// set the page name
|
||||||
$this->page_name = \CoreLibs\Get\System::getPageName();
|
$this->page_name = \CoreLibs\Get\System::getPageName();
|
||||||
|
|
||||||
// set the action ids
|
// NOTE: if any of the "action" vars are used somewhere, it is recommended to NOT set them here
|
||||||
foreach ($this->action_list as $_action) {
|
if ($init_action_vars) {
|
||||||
$this->$_action = $_POST[$_action] ?? '';
|
$this->adbSetActionVars();
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($set_default_acl_level === null) {
|
if ($set_default_acl_level === null) {
|
||||||
@@ -170,9 +189,12 @@ class Backend
|
|||||||
}
|
}
|
||||||
|
|
||||||
// queue key
|
// queue key
|
||||||
if (preg_match("/^(add|save|delete|remove|move|up|down|push_live)$/", $this->action)) {
|
if (preg_match("/^(add|save|delete|remove|move|up|down|push_live)$/", $this->action ?? '')) {
|
||||||
$this->queue_key = \CoreLibs\Create\RandomKey::randomKeyGen(3);
|
$this->queue_key = \CoreLibs\Create\RandomKey::randomKeyGen(3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// check what edit log data write types are allowed
|
||||||
|
$this->adbSetEditLogWriteTypeAvailable();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -183,7 +205,26 @@ class Backend
|
|||||||
// NO OP
|
// NO OP
|
||||||
}
|
}
|
||||||
|
|
||||||
// PUBLIC METHODS |=================================================>
|
// MARK: PRIVATE METHODS
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the write types that are allowed
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function adbSetEditLogWriteTypeAvailable()
|
||||||
|
{
|
||||||
|
// check what edit log data write types are allowed
|
||||||
|
$this->write_types_available = self::WRITE_TYPES;
|
||||||
|
if (!function_exists('bzcompress')) {
|
||||||
|
$this->write_types_available = array_diff($this->write_types_available, ['BINARY', 'BZIP']);
|
||||||
|
}
|
||||||
|
if (!function_exists('gzcompress')) {
|
||||||
|
$this->write_types_available = array_diff($this->write_types_available, ['LZIP']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: PUBLIC METHODS |=================================================>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* set internal ACL from login ACL
|
* set internal ACL from login ACL
|
||||||
@@ -195,30 +236,117 @@ class Backend
|
|||||||
$this->acl = $acl;
|
$this->acl = $acl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return current set ACL
|
||||||
|
*
|
||||||
|
* @return array<mixed>
|
||||||
|
*/
|
||||||
|
public function adbGetAcl(): array
|
||||||
|
{
|
||||||
|
return $this->acl;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set _POST action vars if needed
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function adbSetActionVars()
|
||||||
|
{
|
||||||
|
// set the action ids
|
||||||
|
foreach ($this->action_list as $_action) {
|
||||||
|
$this->$_action = $_POST[$_action] ?? '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* return all the action data, if not set, sets entry to null
|
||||||
|
*
|
||||||
|
* @return array{action:?string,action_id:null|string|int,action_sub_id:null|string|int,action_yes:null|string|int|bool,action_flag:?string,action_menu:?string,action_loaded:?string,action_value:?string,action_type:?string,action_error:?string}
|
||||||
|
*/
|
||||||
|
public function adbGetActionSet(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'action' => $this->action ?? null,
|
||||||
|
'action_id' => $this->action_id ?? null,
|
||||||
|
'action_sub_id' => $this->action_sub_id ?? null,
|
||||||
|
'action_yes' => $this->action_yes ?? null,
|
||||||
|
'action_flag' => $this->action_flag ?? null,
|
||||||
|
'action_menu' => $this->action_menu ?? null,
|
||||||
|
'action_loaded' => $this->action_loaded ?? null,
|
||||||
|
'action_value' => $this->action_value ?? null,
|
||||||
|
'action_type' => $this->action_type ?? null,
|
||||||
|
'action_error' => $this->action_error ?? null,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* writes all action vars plus other info into edit_log table
|
* writes all action vars plus other info into edit_log table
|
||||||
*
|
*
|
||||||
* @param string $event any kind of event description,
|
* @param string $event [default=''] any kind of event description,
|
||||||
* @param string|array<mixed> $data any kind of data related to that event
|
* @param string|array<mixed> $data [default=''] any kind of data related to that event
|
||||||
* @param string $write_type write type can bei STRING or BINARY
|
* @param string $write_type [default=JSON] write type can be
|
||||||
* @param string|null $db_schema override target schema
|
* JSON, STRING/SERIEAL, BINARY/BZIP or ZLIB
|
||||||
|
* @param string|null $db_schema [default=null] override target schema
|
||||||
* @return void
|
* @return void
|
||||||
|
* @deprecated Use $login->writeLog() and set action_set from ->adbGetActionSet()
|
||||||
*/
|
*/
|
||||||
public function adbEditLog(
|
public function adbEditLog(
|
||||||
string $event = '',
|
string $event = '',
|
||||||
string|array $data = '',
|
string|array $data = '',
|
||||||
string $write_type = 'STRING',
|
string $write_type = 'JSON',
|
||||||
?string $db_schema = null
|
?string $db_schema = null
|
||||||
): void {
|
): void {
|
||||||
$data_binary = '';
|
$data_binary = '';
|
||||||
$data_write = '';
|
$data_write = '';
|
||||||
if ($write_type == 'BINARY') {
|
// check if write type is valid, if not fallback to JSON
|
||||||
$data_binary = $this->db->dbEscapeBytea((string)bzcompress(serialize($data)));
|
if (!in_array($write_type, $this->write_types_available)) {
|
||||||
$data_write = 'see bzip compressed data_binary field';
|
$this->log->warning('Write type not in allowed array, fallback to JSON', context:[
|
||||||
|
"write_type" => $write_type,
|
||||||
|
"write_list" => $this->write_types_available,
|
||||||
|
]);
|
||||||
|
$write_type = 'JSON';
|
||||||
}
|
}
|
||||||
if ($write_type == 'STRING') {
|
switch ($write_type) {
|
||||||
$data_binary = '';
|
case 'BINARY':
|
||||||
$data_write = $this->db->dbEscapeString(serialize($data));
|
case 'BZIP':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea((string)bzcompress(serialize($data)));
|
||||||
|
$data_write = Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'BZIP',
|
||||||
|
'message' => 'see bzip compressed data_binary field'
|
||||||
|
]);
|
||||||
|
break;
|
||||||
|
case 'ZLIB':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea((string)gzcompress(serialize($data)));
|
||||||
|
$data_write = Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'ZLIB',
|
||||||
|
'message' => 'see zlib compressed data_binary field'
|
||||||
|
]);
|
||||||
|
break;
|
||||||
|
case 'STRING':
|
||||||
|
case 'SERIAL':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea(Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'SERIAL',
|
||||||
|
'message' => 'see serial string data field'
|
||||||
|
]));
|
||||||
|
$data_write = serialize($data);
|
||||||
|
break;
|
||||||
|
case 'JSON':
|
||||||
|
$data_binary = $this->db->dbEscapeBytea(Json::jsonConvertArrayTo([
|
||||||
|
'type' => 'JSON',
|
||||||
|
'message' => 'see json string data field'
|
||||||
|
]));
|
||||||
|
// must be converted to array
|
||||||
|
if (!is_array($data)) {
|
||||||
|
$data = ["data" => $data];
|
||||||
|
}
|
||||||
|
$data_write = Json::jsonConvertArrayTo($data);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$this->log->alert('Invalid type for data compression was set', context:[
|
||||||
|
"write_type" => $write_type
|
||||||
|
]);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @var string $DB_SCHEMA check schema */
|
/** @var string $DB_SCHEMA check schema */
|
||||||
@@ -228,44 +356,69 @@ class Backend
|
|||||||
} elseif (!empty($this->db->dbGetSchema())) {
|
} elseif (!empty($this->db->dbGetSchema())) {
|
||||||
$DB_SCHEMA = $this->db->dbGetSchema();
|
$DB_SCHEMA = $this->db->dbGetSchema();
|
||||||
}
|
}
|
||||||
$q = "INSERT INTO " . $DB_SCHEMA . ".edit_log "
|
$q = <<<SQL
|
||||||
. "(euid, event_date, event, data, data_binary, page, "
|
INSERT INTO {DB_SCHEMA}.edit_log (
|
||||||
. "ip, user_agent, referer, script_name, query_string, server_name, http_host, "
|
username, euid, ecuid, ecuuid, event_date, event, error, data, data_binary, page,
|
||||||
. "http_accept, http_accept_charset, http_accept_encoding, session_id, "
|
ip, user_agent, referer, script_name, query_string, server_name, http_host,
|
||||||
. "action, action_id, action_yes, action_flag, action_menu, action_loaded, action_value, action_error) "
|
http_accept, http_accept_charset, http_accept_encoding, session_id,
|
||||||
. "VALUES "
|
action, action_id, action_sub_id, action_yes, action_flag, action_menu, action_loaded,
|
||||||
. "(" . $this->db->dbEscapeString(isset($_SESSION['EUID']) && is_numeric($_SESSION['EUID']) ?
|
action_value, action_type, action_error
|
||||||
$_SESSION['EUID'] :
|
) VALUES (
|
||||||
'NULL')
|
$1, $2, $3, $4, NOW(), $5, $6, $7, $8, $9,
|
||||||
. ", "
|
$10, $11, $12, $13, $14, $15, $16,
|
||||||
. "NOW(), "
|
$17, $18, $19, $20,
|
||||||
. "'" . $this->db->dbEscapeString((string)$event) . "', "
|
$21, $22, $23, $24, $25, $26, $27,
|
||||||
. "'" . $data_write . "', "
|
$28, $29, $30
|
||||||
. "'" . $data_binary . "', "
|
)
|
||||||
. "'" . $this->db->dbEscapeString((string)$this->page_name) . "', "
|
SQL;
|
||||||
. "'" . ($_SERVER["REMOTE_ADDR"] ?? '') . "', "
|
$this->db->dbExecParams(
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['HTTP_USER_AGENT'] ?? '') . "', "
|
str_replace(
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['HTTP_REFERER'] ?? '') . "', "
|
['{DB_SCHEMA}'],
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['SCRIPT_FILENAME'] ?? '') . "', "
|
[$DB_SCHEMA],
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['QUERY_STRING'] ?? '') . "', "
|
$q
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['SERVER_NAME'] ?? '') . "', "
|
),
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['HTTP_HOST'] ?? '') . "', "
|
[
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['HTTP_ACCEPT'] ?? '') . "', "
|
// row 1
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['HTTP_ACCEPT_CHARSET'] ?? '') . "', "
|
'',
|
||||||
. "'" . $this->db->dbEscapeString($_SERVER['HTTP_ACCEPT_ENCODING'] ?? '') . "', "
|
is_numeric($this->session->get('EUID')) ?
|
||||||
. ($this->session->getSessionId() === false ?
|
$this->session->get('EUID') : null,
|
||||||
"NULL" :
|
is_string($this->session->get('ECUID')) ?
|
||||||
"'" . $this->session->getSessionId() . "'")
|
$this->session->get('ECUID') : null,
|
||||||
. ", "
|
!empty($this->session->get('ECUUID')) && Uids::validateUuuidv4($this->session->get('ECUID')) ?
|
||||||
. "'" . $this->db->dbEscapeString($this->action) . "', "
|
$this->session->get('ECUID') : null,
|
||||||
. "'" . $this->db->dbEscapeString($this->action_id) . "', "
|
(string)$event,
|
||||||
. "'" . $this->db->dbEscapeString($this->action_yes) . "', "
|
'',
|
||||||
. "'" . $this->db->dbEscapeString($this->action_flag) . "', "
|
$data_write,
|
||||||
. "'" . $this->db->dbEscapeString($this->action_menu) . "', "
|
$data_binary,
|
||||||
. "'" . $this->db->dbEscapeString($this->action_loaded) . "', "
|
(string)$this->page_name,
|
||||||
. "'" . $this->db->dbEscapeString($this->action_value) . "', "
|
// row 2
|
||||||
. "'" . $this->db->dbEscapeString($this->action_error) . "')";
|
$_SERVER["REMOTE_ADDR"] ?? '',
|
||||||
$this->db->dbExec($q, 'NULL');
|
$_SERVER['HTTP_USER_AGENT'] ?? '',
|
||||||
|
$_SERVER['HTTP_REFERER'] ?? '',
|
||||||
|
$_SERVER['SCRIPT_FILENAME'] ?? '',
|
||||||
|
$_SERVER['QUERY_STRING'] ?? '',
|
||||||
|
$_SERVER['SERVER_NAME'] ?? '',
|
||||||
|
$_SERVER['HTTP_HOST'] ?? '',
|
||||||
|
// row 3
|
||||||
|
$_SERVER['HTTP_ACCEPT'] ?? '',
|
||||||
|
$_SERVER['HTTP_ACCEPT_CHARSET'] ?? '',
|
||||||
|
$_SERVER['HTTP_ACCEPT_ENCODING'] ?? '',
|
||||||
|
$this->session->getSessionId() !== '' ?
|
||||||
|
$this->session->getSessionId() : null,
|
||||||
|
// row 4
|
||||||
|
$this->action ?? '',
|
||||||
|
$this->action_id ?? '',
|
||||||
|
$this->action_sub_id ?? '',
|
||||||
|
$this->action_yes ?? '',
|
||||||
|
$this->action_flag ?? '',
|
||||||
|
$this->action_menu ?? '',
|
||||||
|
$this->action_loaded ?? '',
|
||||||
|
$this->action_value ?? '',
|
||||||
|
$this->action_type ?? '',
|
||||||
|
$this->action_error ?? '',
|
||||||
|
],
|
||||||
|
'NULL'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -302,10 +455,7 @@ class Backend
|
|||||||
?string $set_content_path = null,
|
?string $set_content_path = null,
|
||||||
int $flag = 0,
|
int $flag = 0,
|
||||||
): array {
|
): array {
|
||||||
if (
|
if ($set_content_path === null) {
|
||||||
$set_content_path === null ||
|
|
||||||
!is_string($set_content_path)
|
|
||||||
) {
|
|
||||||
/** @deprecated adbTopMenu missing set_content_path parameter */
|
/** @deprecated adbTopMenu missing set_content_path parameter */
|
||||||
trigger_error(
|
trigger_error(
|
||||||
'Calling adbTopMenu without set_content_path parameter is deprecated',
|
'Calling adbTopMenu without set_content_path parameter is deprecated',
|
||||||
@@ -318,7 +468,7 @@ class Backend
|
|||||||
}
|
}
|
||||||
|
|
||||||
// get the session pages array
|
// get the session pages array
|
||||||
$PAGES = $_SESSION['PAGES'] ?? null;
|
$PAGES = $this->session->get('PAGES');
|
||||||
if (!isset($PAGES) || !is_array($PAGES)) {
|
if (!isset($PAGES) || !is_array($PAGES)) {
|
||||||
$PAGES = [];
|
$PAGES = [];
|
||||||
}
|
}
|
||||||
@@ -504,9 +654,9 @@ class Backend
|
|||||||
string $data,
|
string $data,
|
||||||
string $key_name,
|
string $key_name,
|
||||||
string $key_value,
|
string $key_value,
|
||||||
string $associate = null,
|
?string $associate = null,
|
||||||
string $file = null,
|
?string $file = null,
|
||||||
string $db_schema = null,
|
?string $db_schema = null,
|
||||||
): void {
|
): void {
|
||||||
/** @var string $DB_SCHEMA check schema */
|
/** @var string $DB_SCHEMA check schema */
|
||||||
$DB_SCHEMA = 'public';
|
$DB_SCHEMA = 'public';
|
||||||
@@ -515,16 +665,30 @@ class Backend
|
|||||||
} elseif (!empty($this->db->dbGetSchema())) {
|
} elseif (!empty($this->db->dbGetSchema())) {
|
||||||
$DB_SCHEMA = $this->db->dbGetSchema();
|
$DB_SCHEMA = $this->db->dbGetSchema();
|
||||||
}
|
}
|
||||||
$q = "INSERT INTO " . $DB_SCHEMA . ".live_queue ("
|
$q = <<<SQL
|
||||||
. "queue_key, key_value, key_name, type, target, data, group_key, action, associate, file"
|
INSERT INTO {DB_SCHEMA}.live_queue (
|
||||||
. ") VALUES ("
|
queue_key, key_value, key_name, type,
|
||||||
. "'" . $this->db->dbEscapeString($queue_key) . "', '" . $this->db->dbEscapeString($key_value) . "', "
|
target, data, group_key, action, associate, file
|
||||||
. "'" . $this->db->dbEscapeString($key_name) . "', '" . $this->db->dbEscapeString($type) . "', "
|
) VALUES (
|
||||||
. "'" . $this->db->dbEscapeString($target) . "', '" . $this->db->dbEscapeString($data) . "', "
|
$1, $2, $3, $4,
|
||||||
. "'" . $this->queue_key . "', '" . $this->action . "', "
|
$5, $6, $7, $8, $9, $10
|
||||||
. "'" . $this->db->dbEscapeString((string)$associate) . "', "
|
)
|
||||||
. "'" . $this->db->dbEscapeString((string)$file) . "')";
|
SQL;
|
||||||
$this->db->dbExec($q);
|
// $this->db->dbExec($q);
|
||||||
|
$this->db->dbExecParams(
|
||||||
|
str_replace(
|
||||||
|
['{DB_SCHEMA}'],
|
||||||
|
[$DB_SCHEMA],
|
||||||
|
$q
|
||||||
|
),
|
||||||
|
[
|
||||||
|
$queue_key, $key_value,
|
||||||
|
$key_name, $type,
|
||||||
|
$target, $data,
|
||||||
|
$this->queue_key, $this->action,
|
||||||
|
(string)$associate, (string)$file
|
||||||
|
]
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ class EditBase
|
|||||||
* construct form generator
|
* construct form generator
|
||||||
*
|
*
|
||||||
* phpcs:ignore
|
* phpcs:ignore
|
||||||
* @param array{db_name:string,db_user:string,db_pass:string,db_host:string,db_port:int,db_schema:string,db_encoding:string,db_type:string,db_ssl:string,db_convert_type?:string[]} $db_config db config array, mandatory
|
* @param array{db_name:string,db_user:string,db_pass:string,db_host:string,db_port:int,db_schema:string,db_encoding:string,db_type:string,db_ssl:string,db_convert_type?:string[],db_convert_placeholder?:bool,db_convert_placeholder_target?:string,db_debug_replace_placeholder?:bool} $db_config db config array, mandatory
|
||||||
* @param \CoreLibs\Logging\Logging $log Logging class, null auto set
|
* @param \CoreLibs\Logging\Logging $log Logging class, null auto set
|
||||||
* @param \CoreLibs\Language\L10n $l10n l10n language class, null auto set
|
* @param \CoreLibs\Language\L10n $l10n l10n language class, null auto set
|
||||||
* @param \CoreLibs\ACL\Login $login login class for ACL settings
|
* @param \CoreLibs\ACL\Login $login login class for ACL settings
|
||||||
|
|||||||
114
src/Basic.php
114
src/Basic.php
@@ -90,7 +90,7 @@ class Basic
|
|||||||
* @deprecated DO NOT USE Class\Basic anymore. Use dedicated logger and sub classes
|
* @deprecated DO NOT USE Class\Basic anymore. Use dedicated logger and sub classes
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(
|
||||||
\CoreLibs\Logging\Logging $log = null,
|
?\CoreLibs\Logging\Logging $log = null,
|
||||||
?string $session_name = null
|
?string $session_name = null
|
||||||
) {
|
) {
|
||||||
trigger_error('Class \CoreLibs\Basic is deprected', E_USER_DEPRECATED);
|
trigger_error('Class \CoreLibs\Basic is deprected', E_USER_DEPRECATED);
|
||||||
@@ -1139,118 +1139,6 @@ class Basic
|
|||||||
|
|
||||||
// *** BETTER PASSWORD OPTIONS END ***
|
// *** BETTER PASSWORD OPTIONS END ***
|
||||||
|
|
||||||
// *** COLORS ***
|
|
||||||
// [!!! DEPRECATED !!!]
|
|
||||||
// moved to \CoreLibs\Convert\Colors
|
|
||||||
|
|
||||||
/**
|
|
||||||
* converts a hex RGB color to the int numbers
|
|
||||||
* @param string $hexStr RGB hexstring
|
|
||||||
* @param bool $returnAsString flag to return as string
|
|
||||||
* @param string $seperator string seperator: default: ","
|
|
||||||
* @return string|array<mixed>|bool false on error or array with RGB or
|
|
||||||
* a string with the seperator
|
|
||||||
* @deprecated use \CoreLibs\Convert\Colors::hex2rgb() instead
|
|
||||||
*/
|
|
||||||
public static function hex2rgb(string $hexStr, bool $returnAsString = false, string $seperator = ',')
|
|
||||||
{
|
|
||||||
trigger_error('Method ' . __METHOD__ . ' is deprecated, use \CoreLibs\Convert\Colors::hex2rgb()', E_USER_DEPRECATED);
|
|
||||||
return \CoreLibs\Convert\Colors::hex2rgb($hexStr, $returnAsString, $seperator);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* converts the rgb values from int data to the valid rgb html hex string
|
|
||||||
* optional can turn of leading #
|
|
||||||
* @param int $red red 0-255
|
|
||||||
* @param int $green green 0-255
|
|
||||||
* @param int $blue blue 0-255
|
|
||||||
* @param bool $hex_prefix default true, prefix with "#"
|
|
||||||
* @return string|bool rgb in hex values with leading # if set
|
|
||||||
* @deprecated use \CoreLibs\Convert\Colors::rgb2hex() instead
|
|
||||||
*/
|
|
||||||
public static function rgb2hex(int $red, int $green, int $blue, bool $hex_prefix = true)
|
|
||||||
{
|
|
||||||
trigger_error('Method ' . __METHOD__ . ' is deprecated, use \CoreLibs\Convert\Colors::rgb2hex()', E_USER_DEPRECATED);
|
|
||||||
return \CoreLibs\Convert\Colors::rgb2hex($red, $green, $blue, $hex_prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* converts and int RGB to the HTML color string in hex format
|
|
||||||
* @param int $red red 0-255
|
|
||||||
* @param int $green green 0-255
|
|
||||||
* @param int $blue blue 0-255
|
|
||||||
* @return string|bool hex rgb string
|
|
||||||
* @deprecated use rgb2hex instead
|
|
||||||
*/
|
|
||||||
public static function rgb2html(int $red, int $green, int $blue)
|
|
||||||
{
|
|
||||||
trigger_error('Method ' . __METHOD__ . ' is deprecated, use \CoreLibs\Convert\Colors::rgb2hex()', E_USER_DEPRECATED);
|
|
||||||
// check that each color is between 0 and 255
|
|
||||||
return \CoreLibs\Convert\Colors::rgb2hex($red, $green, $blue, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* converts RGB to HSB/V values
|
|
||||||
* returns:
|
|
||||||
* array with hue (0-360), sat (0-100%), brightness/value (0-100%)
|
|
||||||
* @param int $red red 0-255
|
|
||||||
* @param int $green green 0-255
|
|
||||||
* @param int $blue blue 0-255
|
|
||||||
* @return array<mixed>|bool Hue, Sat, Brightness/Value
|
|
||||||
* @deprecated use \CoreLibs\Convert\Colors::rgb2hsb() instead
|
|
||||||
*/
|
|
||||||
public static function rgb2hsb(int $red, int $green, int $blue)
|
|
||||||
{
|
|
||||||
trigger_error('Method ' . __METHOD__ . ' is deprecated, use \CoreLibs\Convert\Colors::rgb2hsb()', E_USER_DEPRECATED);
|
|
||||||
return \CoreLibs\Convert\Colors::rgb2hsb($red, $green, $blue);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* converts HSB/V to RGB values RGB is full INT
|
|
||||||
* @param int $H hue 0-360
|
|
||||||
* @param float $S saturation 0-1 (float)
|
|
||||||
* @param float $V brightness/value 0-1 (float)
|
|
||||||
* @return array<mixed>|bool 0 red/1 green/2 blue array
|
|
||||||
* @deprecated use \CoreLibs\Convert\Colors::hsb2rgb() instead
|
|
||||||
*/
|
|
||||||
public static function hsb2rgb(int $H, float $S, float $V)
|
|
||||||
{
|
|
||||||
trigger_error('Method ' . __METHOD__ . ' is deprecated, use \CoreLibs\Convert\Colors::hsb2rgb()', E_USER_DEPRECATED);
|
|
||||||
return \CoreLibs\Convert\Colors::hsb2rgb($H, (int)round($S * 100), (int)round($V * 100));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* converts a RGB (0-255) to HSL
|
|
||||||
* return:
|
|
||||||
* array with hue (0-360), saturation (0-100%) and luminance (0-100%)
|
|
||||||
* @param int $r red 0-255
|
|
||||||
* @param int $g green 0-255
|
|
||||||
* @param int $b blue 0-255
|
|
||||||
* @return array<mixed>|bool hue/sat/luminance
|
|
||||||
* @deprecated use \CoreLibs\Convert\Colors::rgb2hsl() instead
|
|
||||||
*/
|
|
||||||
public static function rgb2hsl(int $r, int $g, int $b)
|
|
||||||
{
|
|
||||||
trigger_error('Method ' . __METHOD__ . ' is deprecated, use \CoreLibs\Convert\Colors::rgb2hsl()', E_USER_DEPRECATED);
|
|
||||||
return \CoreLibs\Convert\Colors::rgb2hsb($r, $g, $b);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* converts an HSL to RGB
|
|
||||||
* @param int $h hue: 0-360 (degrees)
|
|
||||||
* @param float $s saturation: 0-1
|
|
||||||
* @param float $l luminance: 0-1
|
|
||||||
* @return array<mixed>|bool red/blue/green 0-255 each
|
|
||||||
* @deprecated use \CoreLibs\Convert\Colors::hsl2rgb() instead
|
|
||||||
*/
|
|
||||||
public static function hsl2rgb(int $h, float $s, float $l)
|
|
||||||
{
|
|
||||||
trigger_error('Method ' . __METHOD__ . ' is deprecated, use \CoreLibs\Convert\Colors::hsl2rgb()', E_USER_DEPRECATED);
|
|
||||||
return \CoreLibs\Convert\Colors::hsl2rgb($h, $s * 100, $l * 100);
|
|
||||||
}
|
|
||||||
|
|
||||||
// *** COLORS END ***
|
|
||||||
|
|
||||||
// *** EMAIL FUNCTIONS ***
|
// *** EMAIL FUNCTIONS ***
|
||||||
// [!!! DEPRECATED !!!]
|
// [!!! DEPRECATED !!!]
|
||||||
// Moved to \CoreLibs\Check\Email
|
// Moved to \CoreLibs\Check\Email
|
||||||
|
|||||||
@@ -119,6 +119,13 @@ class Colors
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* check if html/css color string is valid
|
* check if html/css color string is valid
|
||||||
|
*
|
||||||
|
* TODO: update check for correct validate values
|
||||||
|
* - space instead of ","
|
||||||
|
* - / opcatiy checks
|
||||||
|
* - loose numeric values
|
||||||
|
* - lab/lch,oklab/oklch validation too
|
||||||
|
*
|
||||||
* @param string $color A color string of any format
|
* @param string $color A color string of any format
|
||||||
* @param int $flags defaults to ALL, else use | to combined from
|
* @param int $flags defaults to ALL, else use | to combined from
|
||||||
* HEX_RGB, HEX_RGBA, RGB, RGBA, HSL, HSLA
|
* HEX_RGB, HEX_RGBA, RGB, RGBA, HSL, HSLA
|
||||||
@@ -168,9 +175,9 @@ class Colors
|
|||||||
if (preg_match("/$regex/", $color)) {
|
if (preg_match("/$regex/", $color)) {
|
||||||
// if valid regex, we now need to check if the content is actually valid
|
// if valid regex, we now need to check if the content is actually valid
|
||||||
// only for rgb/hsl type
|
// only for rgb/hsl type
|
||||||
/** @var int|false */
|
/** @var int<0, max>|false */
|
||||||
$rgb_flag = strpos($color, 'rgb');
|
$rgb_flag = strpos($color, 'rgb');
|
||||||
/** @var int|false */
|
/** @var int<0, max>|false */
|
||||||
$hsl_flag = strpos($color, 'hsl');
|
$hsl_flag = strpos($color, 'hsl');
|
||||||
// if both not match, return true
|
// if both not match, return true
|
||||||
if (
|
if (
|
||||||
|
|||||||
@@ -51,6 +51,23 @@ class File
|
|||||||
// return lines in file
|
// return lines in file
|
||||||
return $lines;
|
return $lines;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get the mime type of a file via finfo
|
||||||
|
* if file not found, throws exception
|
||||||
|
* else returns '' for any other finfo read problem
|
||||||
|
*
|
||||||
|
* @param string $read_file File to read, relative or absolute path
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public static function getMimeType(string $read_file): string
|
||||||
|
{
|
||||||
|
$finfo = new \finfo(FILEINFO_MIME_TYPE);
|
||||||
|
if (!is_file($read_file)) {
|
||||||
|
throw new \UnexpectedValueException('[getMimeType] File not found: ' . $read_file);
|
||||||
|
}
|
||||||
|
return $finfo->file($read_file) ?: '';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -236,6 +236,54 @@ class ArrayHandler
|
|||||||
return $hit_list;
|
return $hit_list;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* main wrapper function for next/prev key
|
||||||
|
*
|
||||||
|
* @param array<mixed> $array array to search in
|
||||||
|
* @param int|string $key key for next/prev
|
||||||
|
* @param bool $next [=true] if to search next or prev
|
||||||
|
* @return int|string|null Next/prev key or null for end/first
|
||||||
|
*/
|
||||||
|
private static function arrayGetKey(array $array, int|string $key, bool $next = true): int|string|null
|
||||||
|
{
|
||||||
|
$keys = array_keys($array);
|
||||||
|
if (($position = array_search($key, $keys, true)) === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
$next_position = $next ? $position + 1 : $position - 1;
|
||||||
|
|
||||||
|
if (!isset($keys[$next_position])) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return $keys[$next_position];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get previous array key from an array
|
||||||
|
* null on not found
|
||||||
|
*
|
||||||
|
* @param array<mixed> $array
|
||||||
|
* @param int|string $key
|
||||||
|
* @return int|string|null Next key, or null for not found
|
||||||
|
*/
|
||||||
|
public static function arrayGetPrevKey(array $array, int|string $key): int|string|null
|
||||||
|
{
|
||||||
|
return self::arrayGetKey($array, $key, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get next array key from an array
|
||||||
|
* null on not found
|
||||||
|
*
|
||||||
|
* @param array<mixed> $array
|
||||||
|
* @param int|string $key
|
||||||
|
* @return int|string|null Next key, or null for not found
|
||||||
|
*/
|
||||||
|
public static function arrayGetNextKey(array $array, int|string $key): int|string|null
|
||||||
|
{
|
||||||
|
return self::arrayGetKey($array, $key, true);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* correctly recursive merges as an array as array_merge_recursive
|
* correctly recursive merges as an array as array_merge_recursive
|
||||||
* just glues things together
|
* just glues things together
|
||||||
@@ -461,6 +509,22 @@ class ArrayHandler
|
|||||||
}
|
}
|
||||||
return $array;
|
return $array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove entries from a simple array, will not keep key order
|
||||||
|
*
|
||||||
|
* any array content is allowed
|
||||||
|
*
|
||||||
|
* https://stackoverflow.com/a/369608
|
||||||
|
*
|
||||||
|
* @param array<mixed> $array Array where elements are located
|
||||||
|
* @param array<mixed> $remove Elements to remove
|
||||||
|
* @return array<mixed> Array with $remove elements removed
|
||||||
|
*/
|
||||||
|
public static function arrayRemoveEntry(array $array, array $remove): array
|
||||||
|
{
|
||||||
|
return array_diff($array, $remove);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -108,7 +108,12 @@ class DateTime
|
|||||||
if (preg_match("/(h|m|s|ms)/", (string)$timestamp)) {
|
if (preg_match("/(h|m|s|ms)/", (string)$timestamp)) {
|
||||||
return (string)$timestamp;
|
return (string)$timestamp;
|
||||||
}
|
}
|
||||||
list($timestamp, $ms) = array_pad(explode('.', (string)round((float)$timestamp, 4)), 2, null);
|
// split to 6 (nano seconds)
|
||||||
|
list($timestamp, $ms) = array_pad(explode('.', (string)round((float)$timestamp, 6)), 2, null);
|
||||||
|
// if micro seconds is on and we have none, set to 0
|
||||||
|
if ($show_micro && $ms === null) {
|
||||||
|
$ms = 0;
|
||||||
|
}
|
||||||
// if negative remember
|
// if negative remember
|
||||||
$negative = false;
|
$negative = false;
|
||||||
if ((int)$timestamp < 0) {
|
if ((int)$timestamp < 0) {
|
||||||
@@ -120,6 +125,10 @@ class DateTime
|
|||||||
$time_string = '';
|
$time_string = '';
|
||||||
// if timestamp is zero, return zero string
|
// if timestamp is zero, return zero string
|
||||||
if ($timestamp == 0) {
|
if ($timestamp == 0) {
|
||||||
|
// if no seconds and we have no microseconds either, show no micro seconds
|
||||||
|
if ($ms == 0) {
|
||||||
|
$ms = null;
|
||||||
|
}
|
||||||
$time_string = '0s';
|
$time_string = '0s';
|
||||||
} else {
|
} else {
|
||||||
for ($i = 0, $iMax = count($timegroups); $i < $iMax; $i++) {
|
for ($i = 0, $iMax = count($timegroups); $i < $iMax; $i++) {
|
||||||
@@ -133,11 +142,8 @@ class DateTime
|
|||||||
}
|
}
|
||||||
// only add ms if we have an ms value
|
// only add ms if we have an ms value
|
||||||
if ($ms !== null) {
|
if ($ms !== null) {
|
||||||
// if we have ms and it has leading zeros, remove them, but only if it is nut just 0
|
// prefix the milliseoncds with 0. and round it max 3 digits and then convert to int
|
||||||
$ms = preg_replace("/^0+(\d+)$/", '${1}', $ms);
|
$ms = round((float)('0.' . $ms), 3) * 1000;
|
||||||
if (!is_string($ms) || empty($ms)) {
|
|
||||||
$ms = '0';
|
|
||||||
}
|
|
||||||
// add ms if there
|
// add ms if there
|
||||||
if ($show_micro) {
|
if ($show_micro) {
|
||||||
$time_string .= ' ' . $ms . 'ms';
|
$time_string .= ' ' . $ms . 'ms';
|
||||||
@@ -151,6 +157,240 @@ class DateTime
|
|||||||
return (string)$time_string;
|
return (string)$time_string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* update timeStringFormat with year and month support
|
||||||
|
*
|
||||||
|
* The following flags have to be set to be timeStringFormat compatible.
|
||||||
|
* Not that on seconds overflow this method will throw an exception, timeStringFormat returned -1s
|
||||||
|
* show_only_days: true,
|
||||||
|
* skip_zero: false,
|
||||||
|
* skip_last_zero: false,
|
||||||
|
* truncate_nanoseconds: true,
|
||||||
|
* truncate_zero_seconds_if_microseconds: false
|
||||||
|
*
|
||||||
|
* @param int|float $seconds Seconds to convert, maxium 6 decimals,
|
||||||
|
* else \UnexpectedValueException will be thrown
|
||||||
|
* if days too large or years too large \LengthException is thrown
|
||||||
|
* @param string $truncate_after [=''] Truncate after which time name, will not round, hard end
|
||||||
|
* values are parts names or interval short names (y, d, f, ...)
|
||||||
|
* if illegal value \UnexpectedValueException is thrown
|
||||||
|
* @param bool $natural_seperator [=false] use ',' and 'and', if off use space
|
||||||
|
* @param bool $name_space_seperator [=false] add a space between the number and the time name
|
||||||
|
* @param bool $show_microseconds [=true] show microseconds
|
||||||
|
* @param bool $short_time_name [=true] use the short time names (eg s instead of seconds)
|
||||||
|
* @param bool $skip_last_zero [=true] skip all trailing zero values, eg 5m 0s => 5m
|
||||||
|
* @param bool $skip_zero [=true] do not show zero values anywhere, eg 1h 0m 20s => 1h 20s
|
||||||
|
* @param bool $show_only_days [=false] do not show years or months, show only days
|
||||||
|
* if truncate after is set to year or month
|
||||||
|
* throws \UnexpectedValueException
|
||||||
|
* @param bool $auto_fix_microseconds [=false] if the micro seconds decimals are more than 6, round them
|
||||||
|
* on defaul throw \UnexpectedValueException
|
||||||
|
* @param bool $truncate_nanoseconds [=false] if microseconds decimals >3 then normal we show 123.4ms
|
||||||
|
* cut the .4 is set to true
|
||||||
|
* @param bool $truncate_zero_seconds_if_microseconds [=true] if we have 0.123 seconds then if true no seconds
|
||||||
|
* will be shown
|
||||||
|
* @return string
|
||||||
|
* @throws \UnexpectedValueException if seconds has more than 6 decimals
|
||||||
|
* if truncate has an illegal value
|
||||||
|
* if truncate is set to year or month and show_only_days is turned on
|
||||||
|
* @throws \LengthException if seconds is too large and show_days_only is selected and days is negetive
|
||||||
|
* or if years is negativ
|
||||||
|
*/
|
||||||
|
public static function intervalStringFormat(
|
||||||
|
int|float $seconds,
|
||||||
|
string $truncate_after = '',
|
||||||
|
bool $natural_seperator = false,
|
||||||
|
bool $name_space_seperator = false,
|
||||||
|
bool $show_microseconds = true,
|
||||||
|
bool $short_time_name = true,
|
||||||
|
bool $skip_last_zero = true,
|
||||||
|
bool $skip_zero = true,
|
||||||
|
bool $show_only_days = false,
|
||||||
|
bool $auto_fix_microseconds = false,
|
||||||
|
bool $truncate_nanoseconds = false,
|
||||||
|
bool $truncate_zero_seconds_if_microseconds = true,
|
||||||
|
): string {
|
||||||
|
// auto fix long seconds, else \UnexpectedValueException will be thrown on error
|
||||||
|
// check if we have float and -> round to 6
|
||||||
|
if ($auto_fix_microseconds === true && is_float($seconds)) {
|
||||||
|
$seconds = round($seconds, 6);
|
||||||
|
}
|
||||||
|
// flag negative + set abs
|
||||||
|
$negative = $seconds < 0 ? '-' : '';
|
||||||
|
$seconds = abs($seconds);
|
||||||
|
// create base time
|
||||||
|
$date_now = new \DateTime("@0");
|
||||||
|
try {
|
||||||
|
$date_seconds = new \DateTime("@$seconds");
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
throw new \UnexpectedValueException(
|
||||||
|
'Seconds value is invalid, too large or more than six decimals: ' . $seconds,
|
||||||
|
1,
|
||||||
|
$e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
$interval = date_diff($date_now, $date_seconds);
|
||||||
|
// if show_only_days and negative but input postive alert that this has to be done in y/m/d ...
|
||||||
|
if ($interval->y < 0) {
|
||||||
|
throw new \LengthException('Input seconds value is too large for years output: ' . $seconds, 2);
|
||||||
|
} elseif ($interval->days < 0 && $show_only_days === true) {
|
||||||
|
throw new \LengthException('Input seconds value is too large for days output: ' . $seconds, 3);
|
||||||
|
}
|
||||||
|
$parts = [
|
||||||
|
'years' => 'y',
|
||||||
|
'months' => 'm',
|
||||||
|
'days' => 'd',
|
||||||
|
'hours' => 'h',
|
||||||
|
'minutes' => 'i',
|
||||||
|
'seconds' => 's',
|
||||||
|
'microseconds' => 'f',
|
||||||
|
];
|
||||||
|
$short_name = [
|
||||||
|
'years' => 'y', 'months' => 'm', 'days' => 'd',
|
||||||
|
'hours' => 'h', 'minutes' => 'm', 'seconds' => 's',
|
||||||
|
'microseconds' => 'ms'
|
||||||
|
];
|
||||||
|
// $skip = false;
|
||||||
|
if (!empty($truncate_after)) {
|
||||||
|
// if truncate after not in key or value in parts
|
||||||
|
if (!in_array($truncate_after, array_keys($parts)) && !in_array($truncate_after, array_values($parts))) {
|
||||||
|
throw new \UnexpectedValueException(
|
||||||
|
'truncate_after has an invalid value: ' . $truncate_after,
|
||||||
|
4
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// if truncate after is y or m and we have show_only_days, throw exception
|
||||||
|
if ($show_only_days === true && in_array($truncate_after, ['y', 'years', 'm', 'months'])) {
|
||||||
|
throw new \UnexpectedValueException(
|
||||||
|
'If show_only_days is turned on, the truncate_after cannot be years or months: '
|
||||||
|
. $truncate_after,
|
||||||
|
5
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// $skip = true;
|
||||||
|
}
|
||||||
|
$formatted = [];
|
||||||
|
$zero_formatted = [];
|
||||||
|
$value_set = false;
|
||||||
|
$add_zero_seconds = false;
|
||||||
|
foreach ($parts as $time_name => $part) {
|
||||||
|
if (
|
||||||
|
// skip for micro seconds
|
||||||
|
($show_microseconds === false && $part == 'f') ||
|
||||||
|
// skip for if days only and we have year or month
|
||||||
|
($show_only_days === true && in_array($part, ['y', 'm']))
|
||||||
|
) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$add_value = 0;
|
||||||
|
if ($show_only_days === true && $part == 'd') {
|
||||||
|
$value = $interval->days;
|
||||||
|
} else {
|
||||||
|
$value = $interval->$part;
|
||||||
|
}
|
||||||
|
// print "-> V: $value | $part, $time_name"
|
||||||
|
// . " | Set: " . ($value_set ? 'Y' : 'N') . ", SkipZ: " . ($skip_zero ? 'Y' : 'N')
|
||||||
|
// . " | SkipLZ: " . ($skip_last_zero ? 'Y' : 'N')
|
||||||
|
// . " | " . ($value != 0 ? 'Not zero' : 'ZERO') . "<br>";
|
||||||
|
if ($value != 0) {
|
||||||
|
if ($part == 'f') {
|
||||||
|
if ($truncate_nanoseconds === true) {
|
||||||
|
$value = round($value, 3);
|
||||||
|
}
|
||||||
|
$value *= 1000;
|
||||||
|
// anything above that is nano seconds?
|
||||||
|
}
|
||||||
|
if ($value) {
|
||||||
|
$value_set = true;
|
||||||
|
}
|
||||||
|
$add_value = 1;
|
||||||
|
} elseif (
|
||||||
|
$value == 0 &&
|
||||||
|
$value_set === true && (
|
||||||
|
$skip_last_zero === false ||
|
||||||
|
$skip_zero === false
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
$add_value = 2;
|
||||||
|
}
|
||||||
|
// echo "ADD VALUE: $add_value<br>";
|
||||||
|
if ($add_value) {
|
||||||
|
// build format
|
||||||
|
$format = "$value";
|
||||||
|
if ($name_space_seperator) {
|
||||||
|
$format .= " ";
|
||||||
|
}
|
||||||
|
if ($short_time_name) {
|
||||||
|
$format .= $short_name[$time_name];
|
||||||
|
} elseif ($value == 1) {
|
||||||
|
$format .= substr($time_name, 0, -1);
|
||||||
|
} else {
|
||||||
|
$format .= $time_name;
|
||||||
|
}
|
||||||
|
if ($add_value == 1) {
|
||||||
|
if (count($zero_formatted) && $skip_zero === false) {
|
||||||
|
$formatted = array_merge($formatted, $zero_formatted);
|
||||||
|
}
|
||||||
|
$zero_formatted = [];
|
||||||
|
$formatted[] = $format;
|
||||||
|
} elseif ($add_value == 2) {
|
||||||
|
$zero_formatted[] = $format;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// if seconds is zero
|
||||||
|
if (
|
||||||
|
$part == 's' && $value == 0 &&
|
||||||
|
$show_microseconds === true &&
|
||||||
|
$truncate_zero_seconds_if_microseconds === false
|
||||||
|
) {
|
||||||
|
$add_zero_seconds = true;
|
||||||
|
}
|
||||||
|
// stop after a truncate is matching
|
||||||
|
if ($part == $truncate_after || $truncate_after == $time_name) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// add all zero entries if we have skip last off
|
||||||
|
if (count($zero_formatted) && $skip_last_zero === false) {
|
||||||
|
$formatted = array_merge($formatted, $zero_formatted);
|
||||||
|
}
|
||||||
|
// print "=> F: " . print_r($formatted, true)
|
||||||
|
// . " | Z: " . print_r($zero_list, true)
|
||||||
|
// . " | ZL: " . print_r($zero_last_list, true)
|
||||||
|
// . "<br>";
|
||||||
|
if (count($formatted) == 0) {
|
||||||
|
// if we have truncate on, then we assume nothing was found
|
||||||
|
if (!empty($truncate_after)) {
|
||||||
|
if (in_array($truncate_after, array_values($parts))) {
|
||||||
|
$truncate_after = array_flip($parts)[$truncate_after];
|
||||||
|
}
|
||||||
|
$time_name = $truncate_after;
|
||||||
|
} else {
|
||||||
|
$time_name = 'seconds';
|
||||||
|
}
|
||||||
|
return '0' . ($name_space_seperator ? ' ' : '')
|
||||||
|
. ($short_time_name ? $short_name[$time_name] : $time_name);
|
||||||
|
} elseif (count($formatted) == 1) {
|
||||||
|
return $negative .
|
||||||
|
($add_zero_seconds ?
|
||||||
|
'0'
|
||||||
|
. ($name_space_seperator ? ' ' : '')
|
||||||
|
. ($short_time_name ? $short_name['seconds'] : 'seconds')
|
||||||
|
. ' '
|
||||||
|
: ''
|
||||||
|
)
|
||||||
|
. $formatted[0];
|
||||||
|
} elseif ($natural_seperator === false) {
|
||||||
|
return $negative . implode(' ', $formatted);
|
||||||
|
} else {
|
||||||
|
$str = implode(', ', array_slice($formatted, 0, -1));
|
||||||
|
if (!empty($formatted[count($formatted) - 1])) {
|
||||||
|
$str .= ' and ' . (string)array_pop($formatted);
|
||||||
|
}
|
||||||
|
return $negative . $str;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* does a reverse of the timeStringFormat and converts the string from
|
* does a reverse of the timeStringFormat and converts the string from
|
||||||
* xd xh xm xs xms to a timestamp.microtime format
|
* xd xh xm xs xms to a timestamp.microtime format
|
||||||
@@ -435,9 +675,9 @@ class DateTime
|
|||||||
foreach ($period as $dt) {
|
foreach ($period as $dt) {
|
||||||
$curr = $dt->format('D');
|
$curr = $dt->format('D');
|
||||||
if ($curr == 'Sat' || $curr == 'Sun') {
|
if ($curr == 'Sat' || $curr == 'Sun') {
|
||||||
$days[2] ++;
|
$days[2]++;
|
||||||
} else {
|
} else {
|
||||||
$days[1] ++;
|
$days[1]++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ($return_named === true) {
|
if ($return_named === true) {
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ class Byte
|
|||||||
* BYTE_FORMAT_ADJUST: sprintf adjusted two 2 decimals
|
* BYTE_FORMAT_ADJUST: sprintf adjusted two 2 decimals
|
||||||
* BYTE_FORMAT_SI: use 1000 instead of 1024
|
* BYTE_FORMAT_SI: use 1000 instead of 1024
|
||||||
* @return string converted byte number (float) with suffix
|
* @return string converted byte number (float) with suffix
|
||||||
* @throws \Exception 1: no valid flag set
|
* @throws \InvalidArgumentException 1: no valid flag set
|
||||||
*/
|
*/
|
||||||
public static function humanReadableByteFormat(string|int|float $bytes, int $flags = 0): string
|
public static function humanReadableByteFormat(string|int|float $bytes, int $flags = 0): string
|
||||||
{
|
{
|
||||||
@@ -63,7 +63,7 @@ class Byte
|
|||||||
$si = false;
|
$si = false;
|
||||||
}
|
}
|
||||||
if ($flags > 7) {
|
if ($flags > 7) {
|
||||||
throw new \Exception("Invalid flags parameter: $flags", 1);
|
throw new \InvalidArgumentException("Invalid flags parameter: $flags", 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
// si or normal
|
// si or normal
|
||||||
@@ -119,7 +119,7 @@ class Byte
|
|||||||
* @param int $flags bitwise flag with use space turned on
|
* @param int $flags bitwise flag with use space turned on
|
||||||
* BYTE_FORMAT_SI: use 1000 instead of 1024
|
* BYTE_FORMAT_SI: use 1000 instead of 1024
|
||||||
* @return string|int|float converted value or original value
|
* @return string|int|float converted value or original value
|
||||||
* @throws \Exception 1: no valid flag set
|
* @throws \InvalidArgumentException 1: no valid flag set
|
||||||
*/
|
*/
|
||||||
public static function stringByteFormat(string|int|float $number, int $flags = 0): string|int|float
|
public static function stringByteFormat(string|int|float $number, int $flags = 0): string|int|float
|
||||||
{
|
{
|
||||||
@@ -130,7 +130,7 @@ class Byte
|
|||||||
$si = false;
|
$si = false;
|
||||||
}
|
}
|
||||||
if ($flags != 0 && $flags != 4) {
|
if ($flags != 0 && $flags != 4) {
|
||||||
throw new \Exception("Invalid flags parameter: $flags", 1);
|
throw new \InvalidArgumentException("Invalid flags parameter: $flags", 1);
|
||||||
}
|
}
|
||||||
// matches in regex
|
// matches in regex
|
||||||
$matches = [];
|
$matches = [];
|
||||||
|
|||||||
359
src/Convert/Color/CieXyz.php
Normal file
359
src/Convert/Color/CieXyz.php
Normal file
@@ -0,0 +1,359 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/12
|
||||||
|
* DESCRIPTION:
|
||||||
|
* CIE XYZ color space conversion
|
||||||
|
* This for various interims work
|
||||||
|
* none public
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Math;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\RGB;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\Lab;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\XYZ;
|
||||||
|
|
||||||
|
class CieXyz
|
||||||
|
{
|
||||||
|
// MARK: public wrapper functions
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert from RGB to OkLab
|
||||||
|
* via xyz D65
|
||||||
|
*
|
||||||
|
* @param RGB $rgb
|
||||||
|
* @return Lab
|
||||||
|
*/
|
||||||
|
public static function rgbViaXyzD65ToOkLab(RGB $rgb): Lab
|
||||||
|
{
|
||||||
|
return self::xyzD65ToOkLab(
|
||||||
|
self::linRgbToXyzD65($rgb)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convet from OkLab to RGB
|
||||||
|
* via xyz D65
|
||||||
|
*
|
||||||
|
* @param Lab $lab
|
||||||
|
* @return RGB
|
||||||
|
*/
|
||||||
|
public static function okLabViaXyzD65ToRgb(Lab $lab): RGB
|
||||||
|
{
|
||||||
|
return self::xyzD65ToLinRgb(
|
||||||
|
self::okLabToXyzD65($lab)
|
||||||
|
)->fromLinear();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert RGB to CIE Lab
|
||||||
|
* via xyz D65 to xyz D50
|
||||||
|
*
|
||||||
|
* @param RGB $rgb
|
||||||
|
* @return Lab
|
||||||
|
*/
|
||||||
|
public static function rgbViaXyzD65ViaXyzD50ToLab(RGB $rgb): Lab
|
||||||
|
{
|
||||||
|
return self::xyzD50ToLab(
|
||||||
|
self::xyzD65ToXyzD50(
|
||||||
|
self::linRgbToXyzD65($rgb)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert CIE Lab to RGB
|
||||||
|
* via xyz D50 to xyz D65
|
||||||
|
*
|
||||||
|
* @param Lab $lab
|
||||||
|
* @return RGB
|
||||||
|
*/
|
||||||
|
public static function labViaXyzD50ViaXyzD65ToRgb(Lab $lab): RGB
|
||||||
|
{
|
||||||
|
return self::xyzD65ToLinRgb(
|
||||||
|
self::xyzD50ToXyxD65(
|
||||||
|
self::labToXyzD50($lab)
|
||||||
|
)
|
||||||
|
)->fromLinear();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert from oklab to cie lab
|
||||||
|
*
|
||||||
|
* @param Lab $lab
|
||||||
|
* @return Lab
|
||||||
|
*/
|
||||||
|
public static function okLabViaXyzD65ViaXyzD50ToLab(Lab $lab): Lab
|
||||||
|
{
|
||||||
|
return self::xyzD50ToLab(
|
||||||
|
self::xyzD65ToXyzD50(
|
||||||
|
self::okLabToXyzD65($lab)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert from cie lab to oklab
|
||||||
|
*
|
||||||
|
* @param Lab $lab
|
||||||
|
* @return Lab
|
||||||
|
*/
|
||||||
|
public static function labViaXyzD50ViaXyzD65ToOkLab(Lab $lab): Lab
|
||||||
|
{
|
||||||
|
return self::xyzD65ToOkLab(
|
||||||
|
self::xyzD50ToXyxD65(
|
||||||
|
self::labToXyzD50($lab)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: helper convert any array to array{float, float, float}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is a hack for phpstan until we write a proper matrix to class
|
||||||
|
* conversion wrapper function
|
||||||
|
*
|
||||||
|
* @param array<array<float|int>|float|int> $_array
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
private static function convertArray(array $_array): array
|
||||||
|
{
|
||||||
|
/** @var array{0:float,1:float,2:float} */
|
||||||
|
return [$_array[0], $_array[1], $_array[2]];
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: xyzD65 <-> xyzD50
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xyzD65 to xyzD50 whitepoint
|
||||||
|
*
|
||||||
|
* @param XYZ $xyz
|
||||||
|
* @return XYZ
|
||||||
|
*/
|
||||||
|
private static function xyzD65ToXyzD50(XYZ $xyz): XYZ
|
||||||
|
{
|
||||||
|
return new XYZ(self::convertArray(Math::multiplyMatrices(
|
||||||
|
a: [
|
||||||
|
[1.0479298208405488, 0.022946793341019088, -0.05019222954313557],
|
||||||
|
[0.029627815688159344, 0.990434484573249, -0.01707382502938514],
|
||||||
|
[-0.009243058152591178, 0.015055144896577895, 0.7518742899580008],
|
||||||
|
],
|
||||||
|
b: $xyz->returnAsArray(),
|
||||||
|
)), options: ["whitepoint" => 'D50']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xyzD50 to xyzD65 whitepoint
|
||||||
|
*
|
||||||
|
* @param XYZ $xyz
|
||||||
|
* @return XYZ
|
||||||
|
*/
|
||||||
|
private static function xyzD50ToXyxD65(XYZ $xyz): XYZ
|
||||||
|
{
|
||||||
|
return new XYZ(self::convertArray(Math::multiplyMatrices(
|
||||||
|
a: [
|
||||||
|
[0.9554734527042182, -0.023098536874261423, 0.0632593086610217],
|
||||||
|
[-0.028369706963208136, 1.0099954580058226, 0.021041398966943008],
|
||||||
|
[0.012314001688319899, -0.020507696433477912, 1.3303659366080753],
|
||||||
|
],
|
||||||
|
b: $xyz->returnAsArray()
|
||||||
|
)), options: ["whitepoint" => 'D65']);
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: xyzD50 <-> Lab
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert xyzD50 to Lab (Cie)
|
||||||
|
*
|
||||||
|
* @param XYZ $xyz
|
||||||
|
* @return Lab
|
||||||
|
*/
|
||||||
|
private static function xyzD50ToLab(XYZ $xyz): Lab
|
||||||
|
{
|
||||||
|
$_xyz = $xyz->returnAsArray();
|
||||||
|
$d50 = [
|
||||||
|
0.3457 / 0.3585,
|
||||||
|
1.00000,
|
||||||
|
(1.0 - 0.3457 - 0.3585) / 0.3585,
|
||||||
|
];
|
||||||
|
|
||||||
|
$a = 216 / 24389;
|
||||||
|
$b = 24389 / 27;
|
||||||
|
|
||||||
|
$_xyz = array_map(
|
||||||
|
fn ($k, $v) => $v / $d50[$k],
|
||||||
|
array_keys($_xyz),
|
||||||
|
array_values($_xyz),
|
||||||
|
);
|
||||||
|
|
||||||
|
$f = array_map(
|
||||||
|
fn ($v) => (($v > $a) ?
|
||||||
|
pow($v, 1 / 3) :
|
||||||
|
(($b * $v + 16) / 116)
|
||||||
|
),
|
||||||
|
$_xyz,
|
||||||
|
);
|
||||||
|
|
||||||
|
return new Lab([
|
||||||
|
(116 * $f[1]) - 16,
|
||||||
|
500 * ($f[0] - $f[1]),
|
||||||
|
200 * ($f[1] - $f[2]),
|
||||||
|
], colorspace: 'CIELab');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert Lab (Cie) to xyz D50
|
||||||
|
*
|
||||||
|
* @param Lab $lab
|
||||||
|
* @return XYZ
|
||||||
|
*/
|
||||||
|
private static function labToXyzD50(Lab $lab): XYZ
|
||||||
|
{
|
||||||
|
$_lab = $lab->returnAsArray();
|
||||||
|
$a = 24389 / 27;
|
||||||
|
$b = 216 / 24389;
|
||||||
|
$f = [];
|
||||||
|
$f[1] = ($_lab[0] + 16) / 116;
|
||||||
|
$f[0] = $_lab[1] / 500 + $f[1];
|
||||||
|
$f[2] = $f[1] - $_lab[2] / 200;
|
||||||
|
$xyz = [
|
||||||
|
// x
|
||||||
|
pow($f[0], 3) > $b ?
|
||||||
|
pow($f[0], 3) :
|
||||||
|
(116 * $f[0] - 16) / $a,
|
||||||
|
// y
|
||||||
|
$_lab[0] > $a * $b ?
|
||||||
|
pow(($_lab[0] + 16) / 116, 3) :
|
||||||
|
$_lab[0] / $a,
|
||||||
|
// z
|
||||||
|
pow($f[2], 3) > $b ?
|
||||||
|
pow($f[2], 3) :
|
||||||
|
(116 * $f[2] - 16) / $a,
|
||||||
|
];
|
||||||
|
|
||||||
|
$d50 = [
|
||||||
|
0.3457 / 0.3585,
|
||||||
|
1.00000,
|
||||||
|
(1.0 - 0.3457 - 0.3585) / 0.3585,
|
||||||
|
];
|
||||||
|
|
||||||
|
return new XYZ(
|
||||||
|
self::convertArray(array_map(
|
||||||
|
fn ($k, $v) => $v * $d50[$k],
|
||||||
|
array_keys($xyz),
|
||||||
|
$xyz,
|
||||||
|
)),
|
||||||
|
options: ["whitepoint" => 'D50']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: xyzD65 <-> (linear)RGB
|
||||||
|
|
||||||
|
/**
|
||||||
|
* convert linear RGB to xyz D65
|
||||||
|
* if rgb is not flagged linear, it will be auto converted
|
||||||
|
*
|
||||||
|
* @param RGB $rgb
|
||||||
|
* @return XYZ
|
||||||
|
*/
|
||||||
|
private static function linRgbToXyzD65(RGB $rgb): XYZ
|
||||||
|
{
|
||||||
|
// if not linear, convert to linear
|
||||||
|
if (!(bool)$rgb->get('linear')) {
|
||||||
|
$rgb = (new RGB($rgb->returnAsArray()))->toLinear();
|
||||||
|
}
|
||||||
|
return new XYZ(self::convertArray(Math::multiplyMatrices(
|
||||||
|
[
|
||||||
|
[0.41239079926595934, 0.357584339383878, 0.1804807884018343],
|
||||||
|
[0.21263900587151027, 0.715168678767756, 0.07219231536073371],
|
||||||
|
[0.01933081871559182, 0.11919477979462598, 0.9505321522496607],
|
||||||
|
],
|
||||||
|
$rgb->returnAsArray()
|
||||||
|
)), options: ["whitepoint" => 'D65']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert xyz D65 to linear RGB
|
||||||
|
*
|
||||||
|
* @param XYZ $xyz
|
||||||
|
* @return RGB
|
||||||
|
*/
|
||||||
|
private static function xyzD65ToLinRgb(XYZ $xyz): RGB
|
||||||
|
{
|
||||||
|
// xyz D65 to linrgb
|
||||||
|
return new RGB(self::convertArray(Math::multiplyMatrices(
|
||||||
|
a : [
|
||||||
|
[ 3.2409699419045226, -1.537383177570094, -0.4986107602930034 ],
|
||||||
|
[ -0.9692436362808796, 1.8759675015077202, 0.04155505740717559 ],
|
||||||
|
[ 0.05563007969699366, -0.20397695888897652, 1.0569715142428786 ],
|
||||||
|
],
|
||||||
|
b : $xyz->returnAsArray()
|
||||||
|
)), options: ["linear" => true]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: xyzD65 <-> OkLab
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xyz D65 to OkLab
|
||||||
|
*
|
||||||
|
* @param XYZ $xyz
|
||||||
|
* @return Lab
|
||||||
|
*/
|
||||||
|
private static function xyzD65ToOkLab(XYZ $xyz): Lab
|
||||||
|
{
|
||||||
|
return new Lab(self::convertArray(Math::multiplyMatrices(
|
||||||
|
[
|
||||||
|
[0.2104542553, 0.7936177850, -0.0040720468],
|
||||||
|
[1.9779984951, -2.4285922050, 0.4505937099],
|
||||||
|
[0.0259040371, 0.7827717662, -0.8086757660],
|
||||||
|
],
|
||||||
|
array_map(
|
||||||
|
callback: fn ($v) => pow((float)$v, 1 / 3),
|
||||||
|
array: Math::multiplyMatrices(
|
||||||
|
a: [
|
||||||
|
[0.8190224432164319, 0.3619062562801221, -0.12887378261216414],
|
||||||
|
[0.0329836671980271, 0.9292868468965546, 0.03614466816999844],
|
||||||
|
[0.048177199566046255, 0.26423952494422764, 0.6335478258136937],
|
||||||
|
],
|
||||||
|
b: $xyz->returnAsArray(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
)), colorspace: 'OkLab');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xyz D65 to OkLab
|
||||||
|
*
|
||||||
|
* @param Lab $lab
|
||||||
|
* @return XYZ
|
||||||
|
*/
|
||||||
|
private static function okLabToXyzD65(Lab $lab): XYZ
|
||||||
|
{
|
||||||
|
return new XYZ(self::convertArray(Math::multiplyMatrices(
|
||||||
|
a: [
|
||||||
|
[1.2268798733741557, -0.5578149965554813, 0.28139105017721583],
|
||||||
|
[-0.04057576262431372, 1.1122868293970594, -0.07171106666151701],
|
||||||
|
[-0.07637294974672142, -0.4214933239627914, 1.5869240244272418],
|
||||||
|
],
|
||||||
|
b: array_map(
|
||||||
|
callback: fn ($v) => is_numeric($v) ? $v ** 3 : 0,
|
||||||
|
array: Math::multiplyMatrices(
|
||||||
|
a: [
|
||||||
|
[0.99999999845051981432, 0.39633779217376785678, 0.21580375806075880339],
|
||||||
|
[1.0000000088817607767, -0.1055613423236563494, -0.063854174771705903402],
|
||||||
|
[1.0000000546724109177, -0.089484182094965759684, -1.2914855378640917399],
|
||||||
|
],
|
||||||
|
// Divide $lightness by 100 to convert from CSS OkLab
|
||||||
|
b: $lab->returnAsArray(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)), options: ["whitepoint" => 'D65']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
1103
src/Convert/Color/Color.php
Normal file
1103
src/Convert/Color/Color.php
Normal file
File diff suppressed because it is too large
Load Diff
190
src/Convert/Color/Coordinates/HSB.php
Normal file
190
src/Convert/Color/Coordinates/HSB.php
Normal file
@@ -0,0 +1,190 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Color Coordinate: HSB/HSV
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Utils;
|
||||||
|
|
||||||
|
class HSB implements Interface\CoordinatesInterface
|
||||||
|
{
|
||||||
|
/** @var array<string> allowed colorspaces */
|
||||||
|
private const COLORSPACES = ['sRGB'];
|
||||||
|
|
||||||
|
/** @var float hue */
|
||||||
|
private float $H = 0.0;
|
||||||
|
/** @var float saturation */
|
||||||
|
private float $S = 0.0;
|
||||||
|
/** @var float brightness / value */
|
||||||
|
private float $B = 0.0;
|
||||||
|
|
||||||
|
/** @var string color space: either ok or cie */
|
||||||
|
private string $colorspace = ''; /** @phpstan-ignore-line */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HSB (HSV) color coordinates
|
||||||
|
* Hue/Saturation/Brightness or Value
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @throws \InvalidArgumentException only array colors allowed
|
||||||
|
*/
|
||||||
|
public function __construct(string|array $colors, string $colorspace = 'sRGB', array $options = [])
|
||||||
|
{
|
||||||
|
if (!is_array($colors)) {
|
||||||
|
throw new \InvalidArgumentException('Only array colors allowed', 0);
|
||||||
|
}
|
||||||
|
$this->setColorspace($colorspace)->parseOptions($options)->setFromArray($colors);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set from array
|
||||||
|
* where 0: Hue, 1: Saturation, 2: Brightness
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(string|array $colors, string $colorspace = 'sRGB', array $options = []): self
|
||||||
|
{
|
||||||
|
return new HSB($colors, $colorspace, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parse options
|
||||||
|
*
|
||||||
|
* @param array<string,string> $options
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function parseOptions(array $options): self
|
||||||
|
{
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param float $value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set(string $name, float $value): void
|
||||||
|
{
|
||||||
|
$name = strtoupper($name);
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
switch ($name) {
|
||||||
|
case 'H':
|
||||||
|
if ($value == 360.0) {
|
||||||
|
$value = 0;
|
||||||
|
}
|
||||||
|
// if ($value < 0 || $value > 360) {
|
||||||
|
if (Utils::compare(0.0, $value, 360.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for hue is not in the range of 0 to 360',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'S':
|
||||||
|
// if ($value < 0 || $value > 100) {
|
||||||
|
if (Utils::compare(0.0, $value, 100.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for saturation is not in the range of 0 to 100',
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'B':
|
||||||
|
// if ($value < 0 || $value > 100) {
|
||||||
|
if (Utils::compare(0.0, $value, 100.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for brightness is not in the range of 0 to 100',
|
||||||
|
3
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$this->$name = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool
|
||||||
|
{
|
||||||
|
$name = strtoupper($name);
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
return $this->$name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the colorspace
|
||||||
|
*
|
||||||
|
* @param string $colorspace
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setColorspace(string $colorspace): self
|
||||||
|
{
|
||||||
|
if (!in_array($colorspace, $this::COLORSPACES)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed colorspace', 0);
|
||||||
|
}
|
||||||
|
$this->colorspace = $colorspace;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: Hue, 1: Saturation, 2: Brightness
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array
|
||||||
|
{
|
||||||
|
return [$this->H, $this->S, $this->B];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color as array
|
||||||
|
* where 0: Hue, 1: Saturation, 2: Brightness
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromArray(array $colors): self
|
||||||
|
{
|
||||||
|
$this->set('H', $colors[0]);
|
||||||
|
$this->set('S', $colors[1]);
|
||||||
|
$this->set('B', $colors[2]);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* no hsb in css
|
||||||
|
*
|
||||||
|
* @param float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
* @throws \ErrorException
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
throw new \ErrorException('HSB is not available as CSS color string', 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
195
src/Convert/Color/Coordinates/HSL.php
Normal file
195
src/Convert/Color/Coordinates/HSL.php
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Color Coordinate: HSL
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Utils;
|
||||||
|
|
||||||
|
class HSL implements Interface\CoordinatesInterface
|
||||||
|
{
|
||||||
|
/** @var array<string> allowed colorspaces */
|
||||||
|
private const COLORSPACES = ['sRGB'];
|
||||||
|
|
||||||
|
/** @var float hue */
|
||||||
|
private float $H = 0.0;
|
||||||
|
/** @var float saturation */
|
||||||
|
private float $S = 0.0;
|
||||||
|
/** @var float lightness (luminance) */
|
||||||
|
private float $L = 0.0;
|
||||||
|
|
||||||
|
/** @var string color space: either sRGB */
|
||||||
|
private string $colorspace = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Color Coordinate HSL
|
||||||
|
* Hue/Saturation/Lightness
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @throws \InvalidArgumentException only array colors allowed
|
||||||
|
*/
|
||||||
|
public function __construct(string|array $colors, string $colorspace = 'sRGB', array $options = [])
|
||||||
|
{
|
||||||
|
if (!is_array($colors)) {
|
||||||
|
throw new \InvalidArgumentException('Only array colors allowed', 0);
|
||||||
|
}
|
||||||
|
$this->setColorspace($colorspace)->parseOptions($options)->setFromArray($colors);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set from array
|
||||||
|
* where 0: Hue, 1: Saturation, 2: Lightness
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(string|array $colors, string $colorspace = 'sRGB', array $options = []): self
|
||||||
|
{
|
||||||
|
return new HSL($colors, $colorspace, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parse options
|
||||||
|
*
|
||||||
|
* @param array<string,string> $options
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function parseOptions(array $options): self
|
||||||
|
{
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param float $value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set(string $name, float $value): void
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
switch ($name) {
|
||||||
|
case 'H':
|
||||||
|
if ($value == 360.0) {
|
||||||
|
$value = 0;
|
||||||
|
}
|
||||||
|
// if ($value < 0 || $value > 360) {
|
||||||
|
if (Utils::compare(0.0, $value, 360.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for hue is not in the range of 0 to 360',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'S':
|
||||||
|
// if ($value < 0 || $value > 100) {
|
||||||
|
if (Utils::compare(0.0, $value, 100.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for saturation is not in the range of 0 to 100',
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'L':
|
||||||
|
// if ($value < 0 || $value > 100) {
|
||||||
|
if (Utils::compare(0.0, $value, 100.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for lightness is not in the range of 0 to 100',
|
||||||
|
3
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$this->$name = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
return $this->$name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the colorspace
|
||||||
|
*
|
||||||
|
* @param string $colorspace
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setColorspace(string $colorspace): self
|
||||||
|
{
|
||||||
|
if (!in_array($colorspace, $this::COLORSPACES)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed colorspace', 0);
|
||||||
|
}
|
||||||
|
$this->colorspace = $colorspace;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: Hue, 1: Saturation, 2: Lightness
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array
|
||||||
|
{
|
||||||
|
return [$this->H, $this->S, $this->L];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color as array
|
||||||
|
* where 0: Hue, 1: Saturation, 2: Lightness
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromArray(array $colors): self
|
||||||
|
{
|
||||||
|
$this->set('H', $colors[0]);
|
||||||
|
$this->set('S', $colors[1]);
|
||||||
|
$this->set('L', $colors[2]);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* convert to css string with optional opacityt
|
||||||
|
*
|
||||||
|
* @param float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
$string = 'hsl('
|
||||||
|
. $this->H
|
||||||
|
. ' '
|
||||||
|
. $this->S
|
||||||
|
. ' '
|
||||||
|
. $this->L
|
||||||
|
. Utils::setOpacity($opacity)
|
||||||
|
. ')';
|
||||||
|
return $string;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
195
src/Convert/Color/Coordinates/HWB.php
Normal file
195
src/Convert/Color/Coordinates/HWB.php
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Color Coordinate: HWB
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Utils;
|
||||||
|
|
||||||
|
class HWB implements Interface\CoordinatesInterface
|
||||||
|
{
|
||||||
|
/** @var array<string> allowed colorspaces */
|
||||||
|
private const COLORSPACES = ['sRGB'];
|
||||||
|
|
||||||
|
/** @var float Hue */
|
||||||
|
private float $H = 0.0;
|
||||||
|
/** @var float Whiteness */
|
||||||
|
private float $W = 0.0;
|
||||||
|
/** @var float Blackness */
|
||||||
|
private float $B = 0.0;
|
||||||
|
|
||||||
|
/** @var string color space: either ok or cie */
|
||||||
|
private string $colorspace = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Color Coordinate: HWB
|
||||||
|
* Hue/Whiteness/Blackness
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @throws \InvalidArgumentException only array colors allowed
|
||||||
|
*/
|
||||||
|
public function __construct(string|array $colors, string $colorspace = 'sRGB', array $options = [])
|
||||||
|
{
|
||||||
|
if (!is_array($colors)) {
|
||||||
|
throw new \InvalidArgumentException('Only array colors allowed', 0);
|
||||||
|
}
|
||||||
|
$this->setColorspace($colorspace)->parseOptions($options)->setFromArray($colors);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set from array
|
||||||
|
* where 0: Hue, 1: Whiteness, 2: Blackness
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(string|array $colors, string $colorspace = 'sRGB', array $options = []): self
|
||||||
|
{
|
||||||
|
return new HWB($colors, $colorspace, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parse options
|
||||||
|
*
|
||||||
|
* @param array<string,string> $options
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function parseOptions(array $options): self
|
||||||
|
{
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param float $value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set(string $name, float $value): void
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
switch ($name) {
|
||||||
|
case 'H':
|
||||||
|
if ($value == 360.0) {
|
||||||
|
$value = 0;
|
||||||
|
}
|
||||||
|
// if ($value < 0 || $value > 360) {
|
||||||
|
if (Utils::compare(0.0, $value, 360.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for hue is not in the range of 0 to 360',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'W':
|
||||||
|
// if ($value < 0 || $value > 100) {
|
||||||
|
if (Utils::compare(0.0, $value, 100.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for whiteness is not in the range of 0 to 100',
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'B':
|
||||||
|
// if ($value < 0 || $value > 100) {
|
||||||
|
if (Utils::compare(0.0, $value, 100.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for blackness is not in the range of 0 to 100',
|
||||||
|
3
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$this->$name = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
return $this->$name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the colorspace
|
||||||
|
*
|
||||||
|
* @param string $colorspace
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setColorspace(string $colorspace): self
|
||||||
|
{
|
||||||
|
if (!in_array($colorspace, $this::COLORSPACES)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed colorspace', 0);
|
||||||
|
}
|
||||||
|
$this->colorspace = $colorspace;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: Hue, 1: Whiteness, 2: Blackness
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array
|
||||||
|
{
|
||||||
|
return [$this->H, $this->W, $this->B];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color as array
|
||||||
|
* where 0: Hue, 1: Whiteness, 2: Blackness
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromArray(array $colors): self
|
||||||
|
{
|
||||||
|
$this->set('H', $colors[0]);
|
||||||
|
$this->set('W', $colors[1]);
|
||||||
|
$this->set('B', $colors[2]);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* convert to css string with optional opacityt
|
||||||
|
*
|
||||||
|
* @param float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
$string = 'hwb('
|
||||||
|
. $this->H
|
||||||
|
. ' '
|
||||||
|
. $this->W
|
||||||
|
. ' '
|
||||||
|
. $this->B
|
||||||
|
. Utils::setOpacity($opacity)
|
||||||
|
. ')';
|
||||||
|
return $string;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: Ymd
|
||||||
|
* DESCRIPTION:
|
||||||
|
* DescriptionHere
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates\Interface;
|
||||||
|
|
||||||
|
interface CoordinatesInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* create class via "Class::create()" call
|
||||||
|
* was used for multiple create interfaces
|
||||||
|
* no longer needed, use "new Class()" instead
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default='']
|
||||||
|
* @param array<string,string|bool|int> $options [default=[]]
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(string|array $colors, string $colorspace = '', array $options = []): self;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: Lightness, 1: a, 2: b
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert into css string with optional opacity
|
||||||
|
*
|
||||||
|
* @param null|float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string;
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
227
src/Convert/Color/Coordinates/LCH.php
Normal file
227
src/Convert/Color/Coordinates/LCH.php
Normal file
@@ -0,0 +1,227 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Color Coordinate: Lch
|
||||||
|
* for oklch or cie
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Utils;
|
||||||
|
|
||||||
|
class LCH implements Interface\CoordinatesInterface
|
||||||
|
{
|
||||||
|
/** @var array<string> allowed colorspaces */
|
||||||
|
private const COLORSPACES = ['OkLab', 'CIELab'];
|
||||||
|
|
||||||
|
/** @var float Lightness/Luminance
|
||||||
|
* CIE: 0 to 100
|
||||||
|
* OKlch: 0.0 to 1.0
|
||||||
|
* BOTH: 0% to 100%
|
||||||
|
*/
|
||||||
|
private float $L = 0.0;
|
||||||
|
/** @var float Chroma
|
||||||
|
* CIE: 0 to 150, cannot be more than 230
|
||||||
|
* OkLch: 0 to 0.4, does not exceed 0.5
|
||||||
|
* BOTH: 0% to 100% (0 to 150, 0 to 0.4)
|
||||||
|
*/
|
||||||
|
private float $C = 0.0;
|
||||||
|
/** @var float Hue
|
||||||
|
* 0 to 360 deg
|
||||||
|
*/
|
||||||
|
private float $H = 0.0;
|
||||||
|
|
||||||
|
/** @var string color space: either ok or cie */
|
||||||
|
private string $colorspace = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Color Coordinate Lch
|
||||||
|
* for oklch
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default='']
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @throws \InvalidArgumentException only array colors allowed
|
||||||
|
*/
|
||||||
|
public function __construct(string|array $colors, string $colorspace = '', array $options = [])
|
||||||
|
{
|
||||||
|
if (!is_array($colors)) {
|
||||||
|
throw new \InvalidArgumentException('Only array colors allowed', 0);
|
||||||
|
}
|
||||||
|
$this->setColorspace($colorspace)->parseOptions($options)->setFromArray($colors);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set from array
|
||||||
|
* where 0: Lightness, 1: Chroma, 2: Hue
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default='']
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(string|array $colors, string $colorspace = '', array $options = []): self
|
||||||
|
{
|
||||||
|
return new LCH($colors, $colorspace, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parse options
|
||||||
|
*
|
||||||
|
* @param array<string,string> $options
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function parseOptions(array $options): self
|
||||||
|
{
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param float $value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set(string $name, float $value): void
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
switch ($name) {
|
||||||
|
case 'L':
|
||||||
|
// if ($this->colorspace == 'CIELab' && ($value < 0 || $value > 100)) {
|
||||||
|
if ($this->colorspace == 'CIELab' && Utils::compare(0.0, $value, 100.0, Utils::ESPILON_BIG)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for lightness is not in the range of 0 to 100 for CIE Lab',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
// } elseif ($this->colorspace == 'OkLab' && ($value < 0 || $value > 1)) {
|
||||||
|
} elseif ($this->colorspace == 'OkLab' && Utils::compare(0.0, $value, 1.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for lightness is not in the range of 0.0 to 1.0 for OkLab',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'C':
|
||||||
|
// if ($this->colorspace == 'CIELab' && ($value < 0 || $value > 230)) {
|
||||||
|
if ($this->colorspace == 'CIELab' && Utils::compare(0.0, $value, 230.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for chroma is not in the range of '
|
||||||
|
. '0 to 150 and a maximum of 230 for CIE Lab',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
// } elseif ($this->colorspace == 'OkLab' && ($value < 0 || $value > 0.55)) {
|
||||||
|
} elseif ($this->colorspace == 'OkLab' && Utils::compare(0.0, $value, 0.55, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for lightness is not in the range of '
|
||||||
|
. '0.0 to 0.4 and a maximum of 0.5 for OkLab',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'H':
|
||||||
|
// if ($value < 0 || $value > 360) {
|
||||||
|
if (Utils::compare(0.0, $value, 360.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for hue is not in the range of 0.0 to 360.0',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$this->$name = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
return $this->$name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the colorspace
|
||||||
|
*
|
||||||
|
* @param string $colorspace
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setColorspace(string $colorspace): self
|
||||||
|
{
|
||||||
|
if (!in_array($colorspace, $this::COLORSPACES)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed colorspace', 0);
|
||||||
|
}
|
||||||
|
$this->colorspace = $colorspace;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: Lightness, 1: Chroma, 2: Hue
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array
|
||||||
|
{
|
||||||
|
return [$this->L, $this->C, $this->H];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color as array
|
||||||
|
* where 0: Lightness, 1: Chroma, 2: Hue
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromArray(array $colors): self
|
||||||
|
{
|
||||||
|
$this->set('L', $colors[0]);
|
||||||
|
$this->set('C', $colors[1]);
|
||||||
|
$this->set('H', $colors[2]);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert into css string with optional opacity
|
||||||
|
*
|
||||||
|
* @param null|float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
$string = '';
|
||||||
|
switch ($this->colorspace) {
|
||||||
|
case 'CIELab':
|
||||||
|
$string = 'lch';
|
||||||
|
break;
|
||||||
|
case 'OkLab':
|
||||||
|
$string = 'oklch';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$string .= '('
|
||||||
|
. $this->L
|
||||||
|
. ' '
|
||||||
|
. $this->C
|
||||||
|
. ' '
|
||||||
|
. $this->H
|
||||||
|
. Utils::setOpacity($opacity)
|
||||||
|
. ');';
|
||||||
|
|
||||||
|
return $string;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
233
src/Convert/Color/Coordinates/Lab.php
Normal file
233
src/Convert/Color/Coordinates/Lab.php
Normal file
@@ -0,0 +1,233 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Color Coordinate: Lab
|
||||||
|
* for oklab or cie
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Utils;
|
||||||
|
|
||||||
|
class Lab implements Interface\CoordinatesInterface
|
||||||
|
{
|
||||||
|
/** @var array<string> allowed colorspaces */
|
||||||
|
private const COLORSPACES = ['OkLab', 'CIELab'];
|
||||||
|
|
||||||
|
/** @var float lightness/luminance
|
||||||
|
* CIE: 0f to 100f
|
||||||
|
* OKlab: 0.0 to 1.0
|
||||||
|
* BOTH: 0% to 100%
|
||||||
|
*/
|
||||||
|
private float $L = 0.0;
|
||||||
|
/** @var float a axis distance
|
||||||
|
* CIE: -125 to 125, cannot be more than +/- 160
|
||||||
|
* OKlab: -0.4 to 0.4, cannot exceed +/- 0.5
|
||||||
|
* BOTH: -100% to 100% (+/-125 or 0.4)
|
||||||
|
*/
|
||||||
|
private float $a = 0.0;
|
||||||
|
/** @var float b axis distance
|
||||||
|
* CIE: -125 to 125, cannot be more than +/- 160
|
||||||
|
* OKlab: -0.4 to 0.4, cannot exceed +/- 0.5
|
||||||
|
* BOTH: -100% to 100% (+/-125 or 0.4)
|
||||||
|
*/
|
||||||
|
private float $b = 0.0;
|
||||||
|
|
||||||
|
/** @var string color space: either ok or cie */
|
||||||
|
private string $colorspace = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Color Coordinate: Lab
|
||||||
|
* for oklab or cie
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default='']
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @throws \InvalidArgumentException only array colors allowed
|
||||||
|
*/
|
||||||
|
public function __construct(string|array $colors, string $colorspace = '', array $options = [])
|
||||||
|
{
|
||||||
|
if (!is_array($colors)) {
|
||||||
|
throw new \InvalidArgumentException('Only array colors allowed', 0);
|
||||||
|
}
|
||||||
|
$this->setColorspace($colorspace)->parseOptions($options)->setFromArray($colors);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set from array
|
||||||
|
* where 0: Lightness, 1: a, 2: b
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default='']
|
||||||
|
* @param array<string,string> $options [default=[]]
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(string|array $colors, string $colorspace = '', array $options = []): self
|
||||||
|
{
|
||||||
|
return new Lab($colors, $colorspace, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parse options
|
||||||
|
*
|
||||||
|
* @param array<string,string> $options
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function parseOptions(array $options): self
|
||||||
|
{
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param float $value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set(string $name, float $value): void
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
switch ($name) {
|
||||||
|
case 'L':
|
||||||
|
// if ($this->colorspace == 'CIELab' && ($value < 0 || $value > 100)) {
|
||||||
|
if ($this->colorspace == 'CIELab' && Utils::compare(0.0, $value, 100.0, Utils::ESPILON_BIG)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for lightness is not in the range of 0 to 100 for CIE Lab',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
// } elseif ($this->colorspace == 'OkLab' && ($value < 0 || $value > 1)) {
|
||||||
|
} elseif ($this->colorspace == 'OkLab' && Utils::compare(0.0, $value, 1.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for lightness is not in the range of 0.0 to 1.0 for OkLab',
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'a':
|
||||||
|
// if ($this->colorspace == 'CIELab' && ($value < -125 || $value > 125)) {
|
||||||
|
if ($this->colorspace == 'CIELab' && Utils::compare(-125.0, $value, 125.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for a is not in the range of -125 to 125 for CIE Lab',
|
||||||
|
2
|
||||||
|
);
|
||||||
|
// } elseif ($this->colorspace == 'OkLab' && ($value < -0.55 || $value > 0.55)) {
|
||||||
|
} elseif ($this->colorspace == 'OkLab' && Utils::compare(-0.55, $value, 0.55, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for a is not in the range of -0.5 to 0.5 for OkLab',
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'b':
|
||||||
|
// if ($this->colorspace == 'CIELab' && ($value < -125 || $value > 125)) {
|
||||||
|
if ($this->colorspace == 'CIELab' && Utils::compare(-125.0, $value, 125.0, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for b is not in the range of -125 to 125 for CIE Lab',
|
||||||
|
3
|
||||||
|
);
|
||||||
|
// } elseif ($this->colorspace == 'OkLab' && ($value < -0.55 || $value > 0.55)) {
|
||||||
|
} elseif ($this->colorspace == 'OkLab' && Utils::compare(-0.55, $value, 0.55, Utils::EPSILON_SMALL)) {
|
||||||
|
throw new \LengthException(
|
||||||
|
'Argument value ' . $value . ' for b is not in the range of -0.5 to 0.5 for OkLab',
|
||||||
|
3
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$this->$name = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
return $this->$name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the colorspace
|
||||||
|
*
|
||||||
|
* @param string $colorspace
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setColorspace(string $colorspace): self
|
||||||
|
{
|
||||||
|
if (!in_array($colorspace, $this::COLORSPACES)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed colorspace', 0);
|
||||||
|
}
|
||||||
|
$this->colorspace = $colorspace;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: Lightness, 1: a, 2: b
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array
|
||||||
|
{
|
||||||
|
return [$this->L, $this->a, $this->b];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color as array
|
||||||
|
* where 0: Lightness, 1: a, 2: b
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromArray(array $colors): self
|
||||||
|
{
|
||||||
|
$this->set('L', $colors[0]);
|
||||||
|
$this->set('a', $colors[1]);
|
||||||
|
$this->set('b', $colors[2]);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert into css string with optional opacity
|
||||||
|
*
|
||||||
|
* @param null|float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
$string = '';
|
||||||
|
switch ($this->colorspace) {
|
||||||
|
case 'CIELab':
|
||||||
|
$string = 'lab';
|
||||||
|
break;
|
||||||
|
case 'OkLab':
|
||||||
|
$string = 'oklab';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$string .= '('
|
||||||
|
. $this->L
|
||||||
|
. ' '
|
||||||
|
. $this->a
|
||||||
|
. ' '
|
||||||
|
. $this->b
|
||||||
|
. Utils::setOpacity($opacity)
|
||||||
|
. ');';
|
||||||
|
|
||||||
|
return $string;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
329
src/Convert/Color/Coordinates/RGB.php
Normal file
329
src/Convert/Color/Coordinates/RGB.php
Normal file
@@ -0,0 +1,329 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Color Coordinate: RGB
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Utils;
|
||||||
|
|
||||||
|
class RGB implements Interface\CoordinatesInterface
|
||||||
|
{
|
||||||
|
/** @var array<string> allowed colorspaces */
|
||||||
|
private const COLORSPACES = ['sRGB'];
|
||||||
|
|
||||||
|
/** @var float red 0 to 255 or 0.0f to 1.0f for linear RGB */
|
||||||
|
private float $R = 0.0;
|
||||||
|
/** @var float green 0 to 255 or 0.0f to 1.0f for linear RGB */
|
||||||
|
private float $G = 0.0;
|
||||||
|
/** @var float blue 0 to 255 or 0.0f to 1.0f for linear RGB */
|
||||||
|
private float $B = 0.0;
|
||||||
|
|
||||||
|
/** @var string color space: either ok or cie */
|
||||||
|
private string $colorspace = '';
|
||||||
|
|
||||||
|
/** @var bool set if this is linear */
|
||||||
|
private bool $linear = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Color Coordinate RGB
|
||||||
|
* @param array{0:float,1:float,2:float}|string $colors RGB color array or hex string
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,bool> $options [default=[]] only "linear" allowed at the moment
|
||||||
|
*/
|
||||||
|
public function __construct(string|array $colors, string $colorspace = 'sRGB', array $options = [])
|
||||||
|
{
|
||||||
|
$this->setColorspace($colorspace)->parseOptions($options);
|
||||||
|
if (is_array($colors)) {
|
||||||
|
$this->setFromArray($colors);
|
||||||
|
} else {
|
||||||
|
$this->setFromHex($colors);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set from array or string
|
||||||
|
* where 0: Red, 1: Green, 2: Blue
|
||||||
|
* OR #ffffff or ffffff
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float}|string $colors RGB color array or hex string
|
||||||
|
* @param string $colorspace [default=sRGB]
|
||||||
|
* @param array<string,bool> $options [default=[]] only "linear" allowed at the moment
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(string|array $colors, string $colorspace = 'sRGB', array $options = []): self
|
||||||
|
{
|
||||||
|
return new RGB($colors, $colorspace, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parse options
|
||||||
|
*
|
||||||
|
* @param array<string,bool> $options
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function parseOptions(array $options): self
|
||||||
|
{
|
||||||
|
$this->flagLinear($options['linear'] ?? false);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param float $value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set(string $name, float $value): void
|
||||||
|
{
|
||||||
|
// do not allow setting linear from outside
|
||||||
|
if ($name == 'linear') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
// if not linear
|
||||||
|
if (!$this->linear && ((int)$value < 0 || (int)$value > 255)) {
|
||||||
|
throw new \LengthException('Argument value ' . $value . ' for color ' . $name
|
||||||
|
. ' is not in the range of 0 to 255', 1);
|
||||||
|
} elseif (
|
||||||
|
// $this->linear && ($value < 0.0 || $value > 1.0)
|
||||||
|
$this->linear && Utils::compare(0.0, $value, 1.0, 0.000001)
|
||||||
|
) {
|
||||||
|
throw new \LengthException('Argument value ' . $value . ' for color ' . $name
|
||||||
|
. ' is not in the range of 0 to 1 for linear rgb', 2);
|
||||||
|
}
|
||||||
|
$this->$name = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float|bool
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
return $this->$name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the colorspace
|
||||||
|
*
|
||||||
|
* @param string $colorspace
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setColorspace(string $colorspace): self
|
||||||
|
{
|
||||||
|
if (!in_array($colorspace, $this::COLORSPACES)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed colorspace', 0);
|
||||||
|
}
|
||||||
|
$this->colorspace = $colorspace;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: Red, 1: Green, 2: Blue
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array
|
||||||
|
{
|
||||||
|
return [$this->R, $this->G, $this->B];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color as array
|
||||||
|
* where 0: Red, 1: Green, 2: Blue
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromArray(array $colors): self
|
||||||
|
{
|
||||||
|
$this->set('R', $colors[0]);
|
||||||
|
$this->set('G', $colors[1]);
|
||||||
|
$this->set('B', $colors[2]);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return current set RGB as hex string. with or without # prefix
|
||||||
|
*
|
||||||
|
* @param bool $hex_prefix
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function returnAsHex(bool $hex_prefix = true): string
|
||||||
|
{
|
||||||
|
// prefix
|
||||||
|
$hex_color = '';
|
||||||
|
if ($hex_prefix === true) {
|
||||||
|
$hex_color = '#';
|
||||||
|
}
|
||||||
|
// convert if in linear
|
||||||
|
if ($this->linear) {
|
||||||
|
$this->fromLinear();
|
||||||
|
}
|
||||||
|
foreach ($this->returnAsArray() as $color) {
|
||||||
|
$hex_color .= str_pad(dechex((int)$color), 2, '0', STR_PAD_LEFT);
|
||||||
|
}
|
||||||
|
return $hex_color;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set colors RGB from hex string
|
||||||
|
*
|
||||||
|
* @param string $hex_string
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromHex(string $hex_string): self
|
||||||
|
{
|
||||||
|
$hex_string = preg_replace("/[^0-9A-Fa-f]/", '', $hex_string); // Gets a proper hex string
|
||||||
|
if (empty($hex_string) || !is_string($hex_string)) {
|
||||||
|
throw new \InvalidArgumentException('hex_string argument cannot be empty', 3);
|
||||||
|
}
|
||||||
|
$rgbArray = [];
|
||||||
|
if (strlen($hex_string) == 6) {
|
||||||
|
// If a proper hex code, convert using bitwise operation.
|
||||||
|
// No overhead... faster
|
||||||
|
$colorVal = hexdec($hex_string);
|
||||||
|
$rgbArray = [
|
||||||
|
0xFF & ($colorVal >> 0x10),
|
||||||
|
0xFF & ($colorVal >> 0x8),
|
||||||
|
0xFF & $colorVal
|
||||||
|
];
|
||||||
|
} elseif (strlen($hex_string) == 3) {
|
||||||
|
// If shorthand notation, need some string manipulations
|
||||||
|
$rgbArray = [
|
||||||
|
hexdec(str_repeat(substr($hex_string, 0, 1), 2)),
|
||||||
|
hexdec(str_repeat(substr($hex_string, 1, 1), 2)),
|
||||||
|
hexdec(str_repeat(substr($hex_string, 2, 1), 2))
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
// Invalid hex color code
|
||||||
|
throw new \UnexpectedValueException('Invalid hex_string: ' . $hex_string, 4);
|
||||||
|
}
|
||||||
|
return $this->setFromArray($rgbArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set as linear
|
||||||
|
* can be used as chain call on create if input is linear RGB
|
||||||
|
* RGB::__construct**(...)->flagLinear();
|
||||||
|
* as it returns self
|
||||||
|
*
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function flagLinear(bool $linear): self
|
||||||
|
{
|
||||||
|
$this->linear = $linear;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Both function source:
|
||||||
|
* https://bottosson.github.io/posts/colorwrong/#what-can-we-do%3F
|
||||||
|
* but reverse f: fromLinear and f_inv for toLinear
|
||||||
|
* Code copied from here:
|
||||||
|
* https://stackoverflow.com/a/12894053
|
||||||
|
*
|
||||||
|
* converts RGB to linear
|
||||||
|
* We come from 0-255 so we need to divide by 255
|
||||||
|
*
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public function toLinear(): self
|
||||||
|
{
|
||||||
|
// if linear, as is
|
||||||
|
if ($this->linear) {
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
$this->flagLinear(true)->setFromArray(array_map(
|
||||||
|
callback: function (int|float $v) {
|
||||||
|
$v = (float)($v / 255);
|
||||||
|
$abs = abs($v);
|
||||||
|
$sign = ($v < 0) ? -1 : 1;
|
||||||
|
return (float)(
|
||||||
|
$abs <= 0.04045 ?
|
||||||
|
$v / 12.92 :
|
||||||
|
$sign * pow(($abs + 0.055) / 1.055, 2.4)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
array: $this->returnAsArray(),
|
||||||
|
));
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* convert back to normal sRGB from linear RGB
|
||||||
|
* we go to 0-255 rgb so we multiply by 255
|
||||||
|
*
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public function fromLinear(): self
|
||||||
|
{
|
||||||
|
// if not linear, as is
|
||||||
|
if (!$this->linear) {
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
$this->flagLinear(false)->setFromArray(array_map(
|
||||||
|
callback: function (int|float $v) {
|
||||||
|
$abs = abs($v);
|
||||||
|
$sign = ($v < 0) ? -1 : 1;
|
||||||
|
// during reverse in some situations the values can become negative in very small ways
|
||||||
|
// like -...E16 and ...E17
|
||||||
|
return ($ret = (float)(255 * (
|
||||||
|
$abs <= 0.0031308 ?
|
||||||
|
$v * 12.92 :
|
||||||
|
$sign * (1.055 * pow($abs, 1.0 / 2.4) - 0.055)
|
||||||
|
))) < 0 ? 0 : $ret;
|
||||||
|
},
|
||||||
|
array: $this->returnAsArray(),
|
||||||
|
));
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* convert to css string with optional opacity
|
||||||
|
* Note: if this is a linear RGB, the data will converted during this operation and the converted back
|
||||||
|
*
|
||||||
|
* @param float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
// if we are in linear mode, convert to normal mode temporary
|
||||||
|
$was_linear = false;
|
||||||
|
if ($this->linear) {
|
||||||
|
$this->fromLinear();
|
||||||
|
$was_linear = true;
|
||||||
|
}
|
||||||
|
$string = 'rgb('
|
||||||
|
. (int)round($this->R, 0)
|
||||||
|
. ' '
|
||||||
|
. (int)round($this->G, 0)
|
||||||
|
. ' '
|
||||||
|
. (int)round($this->B, 0)
|
||||||
|
. Utils::setOpacity($opacity)
|
||||||
|
. ')';
|
||||||
|
if ($was_linear) {
|
||||||
|
$this->toLinear();
|
||||||
|
}
|
||||||
|
return $string;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
202
src/Convert/Color/Coordinates/XYZ.php
Normal file
202
src/Convert/Color/Coordinates/XYZ.php
Normal file
@@ -0,0 +1,202 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Color Coordinate: XYZ (Cie) (colorspace CIEXYZ)
|
||||||
|
* Note, this is only for the D50 & D65 whitepoint conversion
|
||||||
|
* https://en.wikipedia.org/wiki/CIE_1931_color_space#Construction_of_the_CIE_XYZ_color_space_from_the_Wright%E2%80%93Guild_data
|
||||||
|
* https://en.wikipedia.org/wiki/Standard_illuminant#Illuminant_series_D
|
||||||
|
* https://en.wikipedia.org/wiki/Standard_illuminant#D65_values
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
|
// use CoreLibs\Convert\Color\Utils;
|
||||||
|
|
||||||
|
class XYZ implements Interface\CoordinatesInterface
|
||||||
|
{
|
||||||
|
/** @var array<string> allowed colorspaces */
|
||||||
|
private const COLORSPACES = ['CIEXYZ'];
|
||||||
|
/** @var array<string> allowed whitepoints
|
||||||
|
* D50: ICC profile PCS (horizon light) <-> CieLab
|
||||||
|
* D65: RGB color space (noon) <-> linear RGB
|
||||||
|
*/
|
||||||
|
private const ILLUMINANT = ['D50', 'D65'];
|
||||||
|
|
||||||
|
/** @var float X coordinate */
|
||||||
|
private float $X = 0.0;
|
||||||
|
/** @var float Y coordinate (Luminance) */
|
||||||
|
private float $Y = 0.0;
|
||||||
|
/** @var float Z coordinate (blue) */
|
||||||
|
private float $Z = 0.0;
|
||||||
|
|
||||||
|
/** @var string color space: either ok or cie */
|
||||||
|
private string $colorspace = '';
|
||||||
|
|
||||||
|
/** @var string illuminat white point: only D50 and D65 are allowed */
|
||||||
|
private string $whitepoint = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Color Coordinate Lch
|
||||||
|
* for oklch conversion
|
||||||
|
*
|
||||||
|
* @param string|array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=CIEXYZ]
|
||||||
|
* @param array<string,string> $options [default=[]] Only "whitepoint" option allowed
|
||||||
|
* @throws \InvalidArgumentException only array colors allowed
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
string|array $colors,
|
||||||
|
string $colorspace = 'CIEXYZ',
|
||||||
|
array $options = [],
|
||||||
|
) {
|
||||||
|
if (!is_array($colors)) {
|
||||||
|
throw new \InvalidArgumentException('Only array colors allowed', 0);
|
||||||
|
}
|
||||||
|
$this->setColorspace($colorspace)
|
||||||
|
->parseOptions($options)
|
||||||
|
->setFromArray($colors);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set from array
|
||||||
|
* where 0: X, 1: Y, 2: Z
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @param string $colorspace [default=CIEXYZ]
|
||||||
|
* @param array<string,string> $options [default=[]] Only "whitepoint" option allowed
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public static function create(
|
||||||
|
string|array $colors,
|
||||||
|
string $colorspace = 'CIEXYZ',
|
||||||
|
array $options = [],
|
||||||
|
): self {
|
||||||
|
return new XYZ($colors, $colorspace, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parse options
|
||||||
|
*
|
||||||
|
* @param array<string,string> $options
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function parseOptions(array $options): self
|
||||||
|
{
|
||||||
|
$this->setWhitepoint($options['whitepoint'] ?? '');
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param float $value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set(string $name, float $value): void
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
// TODO: setup XYZ value limits
|
||||||
|
// X: 0 to 95.047, Y: 0 to 100, Z: 0 to 108.88
|
||||||
|
// if (Utils::compare(0.0, $value, 100.0, Utils::EPSILON_SMALL))) {
|
||||||
|
// throw new \LengthException('Argument value ' . $value . ' for color ' . $name
|
||||||
|
// . ' is not in the range of 0 to 100.0', 1);
|
||||||
|
// }
|
||||||
|
$this->$name = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get color
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return float
|
||||||
|
*/
|
||||||
|
public function get(string $name): float|string|bool
|
||||||
|
{
|
||||||
|
if (!property_exists($this, $name)) {
|
||||||
|
throw new \ErrorException('Creation of dynamic property is not allowed', 0);
|
||||||
|
}
|
||||||
|
return $this->$name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the colorspace
|
||||||
|
*
|
||||||
|
* @param string $colorspace
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setColorspace(string $colorspace): self
|
||||||
|
{
|
||||||
|
if (!in_array($colorspace, $this::COLORSPACES)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed colorspace', 0);
|
||||||
|
}
|
||||||
|
$this->colorspace = $colorspace;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the whitepoint flag
|
||||||
|
*
|
||||||
|
* @param string $whitepoint
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setWhitepoint(string $whitepoint): self
|
||||||
|
{
|
||||||
|
if (empty($whitepoint)) {
|
||||||
|
$this->whitepoint = '';
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
if (!in_array($whitepoint, $this::ILLUMINANT)) {
|
||||||
|
throw new \InvalidArgumentException('Not allowed whitepoint', 0);
|
||||||
|
}
|
||||||
|
$this->whitepoint = $whitepoint;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the color as array
|
||||||
|
* where 0: X, 1: Y, 2: Z
|
||||||
|
*
|
||||||
|
* @return array{0:float,1:float,2:float}
|
||||||
|
*/
|
||||||
|
public function returnAsArray(): array
|
||||||
|
{
|
||||||
|
return [$this->X, $this->Y, $this->Z];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set color as array
|
||||||
|
* where 0: X, 1: Y, 2: Z
|
||||||
|
*
|
||||||
|
* @param array{0:float,1:float,2:float} $colors
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
private function setFromArray(array $colors): self
|
||||||
|
{
|
||||||
|
$this->set('X', $colors[0]);
|
||||||
|
$this->set('Y', $colors[1]);
|
||||||
|
$this->set('Z', $colors[2]);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* no hsb in css
|
||||||
|
*
|
||||||
|
* @param float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
* @throws \ErrorException
|
||||||
|
*/
|
||||||
|
public function toCssString(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
throw new \ErrorException('XYZ is not available as CSS color string', 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
35
src/Convert/Color/Stringify.php
Normal file
35
src/Convert/Color/Stringify.php
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/11
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Convert color coordinate to CSS string
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\RGB;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\HSL;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\HWB;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\Lab;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates\LCH;
|
||||||
|
|
||||||
|
class Stringify
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* return the CSS string including optional opacity
|
||||||
|
*
|
||||||
|
* @param RGB|Lab|LCH|HSL|HWB $data
|
||||||
|
* @param null|float|string $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public static function toCssString(RGB|Lab|LCH|HSL|HWB $data, null|float|string $opacity): string
|
||||||
|
{
|
||||||
|
return $data->toCssString($opacity);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
56
src/Convert/Color/Utils.php
Normal file
56
src/Convert/Color/Utils.php
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/11/14
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Utils for color
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\Convert\Color;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Math;
|
||||||
|
|
||||||
|
class Utils
|
||||||
|
{
|
||||||
|
/** @var float deviation allowed for valid data checks, small */
|
||||||
|
public const EPSILON_SMALL = 0.000000000001;
|
||||||
|
/** @var float deviation allowed for valid data checks, medium */
|
||||||
|
public const EPSILON_MEDIUM = 0.0000001;
|
||||||
|
/** @var float deviation allowed for valid data checks, big */
|
||||||
|
public const ESPILON_BIG = 0.0001;
|
||||||
|
|
||||||
|
public static function compare(float $lower, float $value, float $upper, float $epslion): bool
|
||||||
|
{
|
||||||
|
if (
|
||||||
|
Math::compareWithEpsilon($value, '<', $lower, $epslion) ||
|
||||||
|
Math::compareWithEpsilon($value, '>', $upper, $epslion)
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build the opactiy sub string part and return it
|
||||||
|
*
|
||||||
|
* @param null|float|string|null $opacity
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public static function setOpacity(null|float|string $opacity = null): string
|
||||||
|
{
|
||||||
|
// set opacity, either a string or float
|
||||||
|
if (is_string($opacity)) {
|
||||||
|
$opacity = ' / ' . $opacity;
|
||||||
|
} elseif ($opacity !== null) {
|
||||||
|
$opacity = ' / ' . $opacity;
|
||||||
|
} else {
|
||||||
|
$opacity = '';
|
||||||
|
}
|
||||||
|
return $opacity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
@@ -17,6 +17,9 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace CoreLibs\Convert;
|
namespace CoreLibs\Convert;
|
||||||
|
|
||||||
|
use CoreLibs\Convert\Color\Color;
|
||||||
|
use CoreLibs\Convert\Color\Coordinates;
|
||||||
|
|
||||||
class Colors
|
class Colors
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
@@ -30,6 +33,7 @@ class Colors
|
|||||||
* @param bool $hex_prefix default true, prefix with "#"
|
* @param bool $hex_prefix default true, prefix with "#"
|
||||||
* @return string rgb in hex values with leading # if set,
|
* @return string rgb in hex values with leading # if set,
|
||||||
* @throws \LengthException If any argument is not in the range of 0~255
|
* @throws \LengthException If any argument is not in the range of 0~255
|
||||||
|
* @deprecated v9.20.0 use: new Coordinates\RGB([$red, $green, $blue]))->returnAsHex(true/false for #)
|
||||||
*/
|
*/
|
||||||
public static function rgb2hex(
|
public static function rgb2hex(
|
||||||
int $red,
|
int $red,
|
||||||
@@ -37,20 +41,7 @@ class Colors
|
|||||||
int $blue,
|
int $blue,
|
||||||
bool $hex_prefix = true
|
bool $hex_prefix = true
|
||||||
): string {
|
): string {
|
||||||
$hex_color = '';
|
return (new Coordinates\RGB([$red, $green, $blue]))->returnAsHex($hex_prefix);
|
||||||
if ($hex_prefix === true) {
|
|
||||||
$hex_color = '#';
|
|
||||||
}
|
|
||||||
foreach (['red', 'green', 'blue'] as $color) {
|
|
||||||
// if not valid, abort
|
|
||||||
if ($$color < 0 || $$color > 255) {
|
|
||||||
throw new \LengthException('Argument value ' . $$color . ' for color ' . $color
|
|
||||||
. ' is not in the range of 0 to 255', 1);
|
|
||||||
}
|
|
||||||
// pad left with 0
|
|
||||||
$hex_color .= str_pad(dechex($$color), 2, '0', STR_PAD_LEFT);
|
|
||||||
}
|
|
||||||
return $hex_color;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -63,32 +54,29 @@ class Colors
|
|||||||
* or a string with the seperator
|
* or a string with the seperator
|
||||||
* @throws \InvalidArgumentException if hex string is empty
|
* @throws \InvalidArgumentException if hex string is empty
|
||||||
* @throws \UnexpectedValueException if the hex string value is not valid
|
* @throws \UnexpectedValueException if the hex string value is not valid
|
||||||
|
* @deprecated v9.20.0 use: new Coordinates\RGB($hex_string) (build string/array from return data)
|
||||||
*/
|
*/
|
||||||
public static function hex2rgb(
|
public static function hex2rgb(
|
||||||
string $hex_string,
|
string $hex_string,
|
||||||
bool $return_as_string = false,
|
bool $return_as_string = false,
|
||||||
string $seperator = ','
|
string $seperator = ','
|
||||||
): string|array {
|
): string|array {
|
||||||
$hex_string = preg_replace("/[^0-9A-Fa-f]/", '', $hex_string); // Gets a proper hex string
|
|
||||||
if (!is_string($hex_string)) {
|
|
||||||
throw new \InvalidArgumentException('hex_string argument cannot be empty', 1);
|
|
||||||
}
|
|
||||||
$rgbArray = [];
|
$rgbArray = [];
|
||||||
if (strlen($hex_string) == 6) {
|
// rewrite to previous r/g/b key output
|
||||||
// If a proper hex code, convert using bitwise operation.
|
foreach ((new Coordinates\RGB($hex_string))->returnAsArray() as $p => $el) {
|
||||||
// No overhead... faster
|
$k = '';
|
||||||
$colorVal = hexdec($hex_string);
|
switch ($p) {
|
||||||
$rgbArray['r'] = 0xFF & ($colorVal >> 0x10);
|
case 0:
|
||||||
$rgbArray['g'] = 0xFF & ($colorVal >> 0x8);
|
$k = 'r';
|
||||||
$rgbArray['b'] = 0xFF & $colorVal;
|
break;
|
||||||
} elseif (strlen($hex_string) == 3) {
|
case 1:
|
||||||
// If shorthand notation, need some string manipulations
|
$k = 'g';
|
||||||
$rgbArray['r'] = hexdec(str_repeat(substr($hex_string, 0, 1), 2));
|
break;
|
||||||
$rgbArray['g'] = hexdec(str_repeat(substr($hex_string, 1, 1), 2));
|
case 2:
|
||||||
$rgbArray['b'] = hexdec(str_repeat(substr($hex_string, 2, 1), 2));
|
$k = 'b';
|
||||||
} else {
|
break;
|
||||||
// Invalid hex color code
|
}
|
||||||
throw new \UnexpectedValueException('Invalid hex_string: ' . $hex_string, 2);
|
$rgbArray[$k] = (int)round($el);
|
||||||
}
|
}
|
||||||
// returns the rgb string or the associative array
|
// returns the rgb string or the associative array
|
||||||
return $return_as_string ? implode($seperator, $rgbArray) : $rgbArray;
|
return $return_as_string ? implode($seperator, $rgbArray) : $rgbArray;
|
||||||
@@ -105,42 +93,16 @@ class Colors
|
|||||||
* @param int $blue blue 0-255
|
* @param int $blue blue 0-255
|
||||||
* @return array<int|float> Hue, Sat, Brightness/Value
|
* @return array<int|float> Hue, Sat, Brightness/Value
|
||||||
* @throws \LengthException If any argument is not in the range of 0~255
|
* @throws \LengthException If any argument is not in the range of 0~255
|
||||||
|
* @deprecated v9.20.0 use: Color::rgbToHsb(...)->returnAsArray() will return float unrounded
|
||||||
*/
|
*/
|
||||||
public static function rgb2hsb(int $red, int $green, int $blue): array
|
public static function rgb2hsb(int $red, int $green, int $blue): array
|
||||||
{
|
{
|
||||||
// check that rgb is from 0 to 255
|
return array_map(
|
||||||
foreach (['red', 'green', 'blue'] as $color) {
|
fn ($v) => (int)round($v),
|
||||||
if ($$color < 0 || $$color > 255) {
|
Color::rgbToHsb(
|
||||||
throw new \LengthException('Argument value ' . $$color . ' for color ' . $color
|
new Coordinates\RGB([$red, $green, $blue])
|
||||||
. ' is not in the range of 0 to 255', 1);
|
)->returnAsArray()
|
||||||
}
|
);
|
||||||
$$color = $$color / 255;
|
|
||||||
}
|
|
||||||
|
|
||||||
$MAX = max($red, $green, $blue);
|
|
||||||
$MIN = min($red, $green, $blue);
|
|
||||||
$HUE = 0;
|
|
||||||
|
|
||||||
if ($MAX == $MIN) {
|
|
||||||
return [0, 0, round($MAX * 100)];
|
|
||||||
}
|
|
||||||
if ($red == $MAX) {
|
|
||||||
$HUE = ($green - $blue) / ($MAX - $MIN);
|
|
||||||
} elseif ($green == $MAX) {
|
|
||||||
$HUE = 2 + (($blue - $red) / ($MAX - $MIN));
|
|
||||||
} elseif ($blue == $MAX) {
|
|
||||||
$HUE = 4 + (($red - $green) / ($MAX - $MIN));
|
|
||||||
}
|
|
||||||
$HUE *= 60;
|
|
||||||
if ($HUE < 0) {
|
|
||||||
$HUE += 360;
|
|
||||||
}
|
|
||||||
|
|
||||||
return [
|
|
||||||
(int)round($HUE),
|
|
||||||
(int)round((($MAX - $MIN) / $MAX) * 100),
|
|
||||||
(int)round($MAX * 100)
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -153,80 +115,16 @@ class Colors
|
|||||||
* @param float $V brightness/value 0-100 (int)
|
* @param float $V brightness/value 0-100 (int)
|
||||||
* @return array<int> 0 red/1 green/2 blue array as 0-255
|
* @return array<int> 0 red/1 green/2 blue array as 0-255
|
||||||
* @throws \LengthException If any argument is not in the valid range
|
* @throws \LengthException If any argument is not in the valid range
|
||||||
|
* @deprecated v9.20.0 use: Color::hsbToRgb(...)->returnAsArray() will return float unrounded
|
||||||
*/
|
*/
|
||||||
public static function hsb2rgb(float $H, float $S, float $V): array
|
public static function hsb2rgb(float $H, float $S, float $V): array
|
||||||
{
|
{
|
||||||
// check that H is 0 to 359, 360 = 0
|
return array_map(
|
||||||
// and S and V are 0 to 1
|
fn ($v) => (int)round($v),
|
||||||
if ($H == 360) {
|
Color::hsbToRgb(
|
||||||
$H = 0;
|
new Coordinates\HSB([$H, $S, $V])
|
||||||
}
|
)->returnAsArray()
|
||||||
if ($H < 0 || $H > 359) {
|
);
|
||||||
throw new \LengthException('Argument value ' . $H . ' for hue is not in the range of 0 to 359', 1);
|
|
||||||
}
|
|
||||||
if ($S < 0 || $S > 100) {
|
|
||||||
throw new \LengthException('Argument value ' . $S . ' for saturation is not in the range of 0 to 100', 2);
|
|
||||||
}
|
|
||||||
if ($V < 0 || $V > 100) {
|
|
||||||
throw new \LengthException('Argument value ' . $V . ' for brightness is not in the range of 0 to 100', 3);
|
|
||||||
}
|
|
||||||
// convert to internal 0-1 format
|
|
||||||
$S /= 100;
|
|
||||||
$V /= 100;
|
|
||||||
|
|
||||||
if ($S == 0) {
|
|
||||||
$V = (int)round($V * 255);
|
|
||||||
return [$V, $V, $V];
|
|
||||||
}
|
|
||||||
|
|
||||||
$Hi = floor($H / 60);
|
|
||||||
$f = ($H / 60) - $Hi;
|
|
||||||
$p = $V * (1 - $S);
|
|
||||||
$q = $V * (1 - ($S * $f));
|
|
||||||
$t = $V * (1 - ($S * (1 - $f)));
|
|
||||||
|
|
||||||
switch ($Hi) {
|
|
||||||
case 0:
|
|
||||||
$red = $V;
|
|
||||||
$green = $t;
|
|
||||||
$blue = $p;
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
$red = $q;
|
|
||||||
$green = $V;
|
|
||||||
$blue = $p;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
$red = $p;
|
|
||||||
$green = $V;
|
|
||||||
$blue = $t;
|
|
||||||
break;
|
|
||||||
case 3:
|
|
||||||
$red = $p;
|
|
||||||
$green = $q;
|
|
||||||
$blue = $V;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
$red = $t;
|
|
||||||
$green = $p;
|
|
||||||
$blue = $V;
|
|
||||||
break;
|
|
||||||
case 5:
|
|
||||||
$red = $V;
|
|
||||||
$green = $p;
|
|
||||||
$blue = $q;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
$red = 0;
|
|
||||||
$green = 0;
|
|
||||||
$blue = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return [
|
|
||||||
(int)round($red * 255),
|
|
||||||
(int)round($green * 255),
|
|
||||||
(int)round($blue * 255)
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -239,50 +137,16 @@ class Colors
|
|||||||
* @param int $blue blue 0-255
|
* @param int $blue blue 0-255
|
||||||
* @return array<float> hue/sat/luminance
|
* @return array<float> hue/sat/luminance
|
||||||
* @throws \LengthException If any argument is not in the range of 0~255
|
* @throws \LengthException If any argument is not in the range of 0~255
|
||||||
|
* @deprecated v9.20.0 use: Color::rgbToHsl(...)->returnAsArray() will return float unrounded
|
||||||
*/
|
*/
|
||||||
public static function rgb2hsl(int $red, int $green, int $blue): array
|
public static function rgb2hsl(int $red, int $green, int $blue): array
|
||||||
{
|
{
|
||||||
// check that rgb is from 0 to 255
|
return array_map(
|
||||||
foreach (['red', 'green', 'blue'] as $color) {
|
fn ($v) => round($v, 1),
|
||||||
if ($$color < 0 || $$color > 255) {
|
Color::rgbToHsl(
|
||||||
throw new \LengthException('Argument value ' . $$color . ' for color ' . $color
|
new Coordinates\RGB([$red, $green, $blue])
|
||||||
. ' is not in the range of 0 to 255', 1);
|
)->returnAsArray()
|
||||||
}
|
);
|
||||||
$$color = $$color / 255;
|
|
||||||
}
|
|
||||||
|
|
||||||
$min = min($red, $green, $blue);
|
|
||||||
$max = max($red, $green, $blue);
|
|
||||||
$chroma = $max - $min;
|
|
||||||
$sat = 0;
|
|
||||||
$hue = 0;
|
|
||||||
// luminance
|
|
||||||
$lum = ($max + $min) / 2;
|
|
||||||
|
|
||||||
// achromatic
|
|
||||||
if ($chroma == 0) {
|
|
||||||
// H, S, L
|
|
||||||
return [0.0, 0.0, round($lum * 100, 1)];
|
|
||||||
} else {
|
|
||||||
$sat = $chroma / (1 - abs(2 * $lum - 1));
|
|
||||||
if ($max == $red) {
|
|
||||||
$hue = fmod((($green - $blue) / $chroma), 6);
|
|
||||||
if ($hue < 0) {
|
|
||||||
$hue = (6 - fmod(abs($hue), 6));
|
|
||||||
}
|
|
||||||
} elseif ($max == $green) {
|
|
||||||
$hue = ($blue - $red) / $chroma + 2;
|
|
||||||
} elseif ($max == $blue) {
|
|
||||||
$hue = ($red - $green) / $chroma + 4;
|
|
||||||
}
|
|
||||||
$hue = $hue * 60;
|
|
||||||
// $sat = 1 - abs(2 * $lum - 1);
|
|
||||||
return [
|
|
||||||
round($hue, 1),
|
|
||||||
round($sat * 100, 1),
|
|
||||||
round($lum * 100, 1)
|
|
||||||
];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -294,57 +158,16 @@ class Colors
|
|||||||
* @param float $lum luminance: 0-100
|
* @param float $lum luminance: 0-100
|
||||||
* @return array<int,float|int> red/blue/green 0-255 each
|
* @return array<int,float|int> red/blue/green 0-255 each
|
||||||
* @throws \LengthException If any argument is not in the valid range
|
* @throws \LengthException If any argument is not in the valid range
|
||||||
|
* @deprecated v9.20.0 use: Color::hslToRgb(...)->returnAsArray() will return float unrounded
|
||||||
*/
|
*/
|
||||||
public static function hsl2rgb(float $hue, float $sat, float $lum): array
|
public static function hsl2rgb(float $hue, float $sat, float $lum): array
|
||||||
{
|
{
|
||||||
if ($hue == 360) {
|
return array_map(
|
||||||
$hue = 0;
|
fn ($v) => round($v),
|
||||||
}
|
Color::hslToRgb(
|
||||||
if ($hue < 0 || $hue > 359) {
|
new Coordinates\HSL([$hue, $sat, $lum])
|
||||||
throw new \LengthException('Argument value ' . $hue . ' for hue is not in the range of 0 to 359', 1);
|
)->returnAsArray()
|
||||||
}
|
);
|
||||||
if ($sat < 0 || $sat > 100) {
|
|
||||||
throw new \LengthException('Argument value ' . $sat . ' for saturation is not in the range of 0 to 100', 2);
|
|
||||||
}
|
|
||||||
if ($lum < 0 || $lum > 100) {
|
|
||||||
throw new \LengthException('Argument value ' . $lum . ' for luminance is not in the range of 0 to 100', 3);
|
|
||||||
}
|
|
||||||
// calc to internal convert value for hue
|
|
||||||
$hue = (1 / 360) * $hue;
|
|
||||||
// convert to internal 0-1 format
|
|
||||||
$sat /= 100;
|
|
||||||
$lum /= 100;
|
|
||||||
// if saturation is 0
|
|
||||||
if ($sat == 0) {
|
|
||||||
$lum = (int)round($lum * 255);
|
|
||||||
return [$lum, $lum, $lum];
|
|
||||||
} else {
|
|
||||||
$m2 = $lum < 0.5 ? $lum * ($sat + 1) : ($lum + $sat) - ($lum * $sat);
|
|
||||||
$m1 = $lum * 2 - $m2;
|
|
||||||
$hueue = function ($base) use ($m1, $m2) {
|
|
||||||
// base = hue, hue > 360 (1) - 360 (1), else < 0 + 360 (1)
|
|
||||||
$base = $base < 0 ? $base + 1 : ($base > 1 ? $base - 1 : $base);
|
|
||||||
// 6: 60, 2: 180, 3: 240
|
|
||||||
// 2/3 = 240
|
|
||||||
// 1/3 = 120 (all from 360)
|
|
||||||
if ($base * 6 < 1) {
|
|
||||||
return $m1 + ($m2 - $m1) * $base * 6;
|
|
||||||
}
|
|
||||||
if ($base * 2 < 1) {
|
|
||||||
return $m2;
|
|
||||||
}
|
|
||||||
if ($base * 3 < 2) {
|
|
||||||
return $m1 + ($m2 - $m1) * ((2 / 3) - $base) * 6;
|
|
||||||
}
|
|
||||||
return $m1;
|
|
||||||
};
|
|
||||||
|
|
||||||
return [
|
|
||||||
(int)round(255 * $hueue($hue + (1 / 3))),
|
|
||||||
(int)round(255 * $hueue($hue)),
|
|
||||||
(int)round(255 * $hueue($hue - (1 / 3)))
|
|
||||||
];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ class SetVarTypeMain
|
|||||||
*/
|
*/
|
||||||
protected static function makeStrMain(
|
protected static function makeStrMain(
|
||||||
mixed $val,
|
mixed $val,
|
||||||
string $default = null,
|
?string $default = null,
|
||||||
bool $to_null = false
|
bool $to_null = false
|
||||||
): ?string {
|
): ?string {
|
||||||
// int/float/string/bool/null, everything else is ignored
|
// int/float/string/bool/null, everything else is ignored
|
||||||
@@ -113,7 +113,7 @@ class SetVarTypeMain
|
|||||||
*/
|
*/
|
||||||
protected static function makeIntMain(
|
protected static function makeIntMain(
|
||||||
mixed $val,
|
mixed $val,
|
||||||
int $default = null,
|
?int $default = null,
|
||||||
bool $to_null = false
|
bool $to_null = false
|
||||||
): ?int {
|
): ?int {
|
||||||
// if we can filter it to a valid int, we can convert it
|
// if we can filter it to a valid int, we can convert it
|
||||||
@@ -167,7 +167,7 @@ class SetVarTypeMain
|
|||||||
*/
|
*/
|
||||||
protected static function makeFloatMain(
|
protected static function makeFloatMain(
|
||||||
mixed $val,
|
mixed $val,
|
||||||
float $default = null,
|
?float $default = null,
|
||||||
bool $to_null = false
|
bool $to_null = false
|
||||||
): ?float {
|
): ?float {
|
||||||
if (
|
if (
|
||||||
|
|||||||
@@ -16,16 +16,22 @@ class Html
|
|||||||
/**
|
/**
|
||||||
* full wrapper for html entities
|
* full wrapper for html entities
|
||||||
*
|
*
|
||||||
|
* uses default params as: ENT_QUOTES | ENT_HTML5
|
||||||
|
* switches from ENT_HTML401 to ENT_HTML5 as we assume all our pages have <!DOCTYPE html>
|
||||||
|
* removed: ENT_SUBSTITUTE -> wrong characters will be replaced with space
|
||||||
|
* encodes in UTF-8
|
||||||
|
* does not double encode
|
||||||
|
*
|
||||||
* @param mixed $string string to html encode
|
* @param mixed $string string to html encode
|
||||||
|
* @param int $flags [default: ENT_QUOTES | ENT_HTML5]
|
||||||
* @return mixed if string, encoded, else as is (eg null)
|
* @return mixed if string, encoded, else as is (eg null)
|
||||||
*/
|
*/
|
||||||
public static function htmlent(mixed $string): mixed
|
public static function htmlent(mixed $string, int $flags = ENT_QUOTES | ENT_HTML5): mixed
|
||||||
{
|
{
|
||||||
if (is_string($string)) {
|
if (is_string($string)) {
|
||||||
return htmlentities($string, ENT_COMPAT | ENT_HTML401, 'UTF-8', false);
|
return htmlentities($string, $flags, 'UTF-8', false);
|
||||||
} else {
|
|
||||||
return $string;
|
|
||||||
}
|
}
|
||||||
|
return $string;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -54,14 +60,10 @@ class Html
|
|||||||
*/
|
*/
|
||||||
public static function checked(array|string $haystack, string $needle, int $type = 0): ?string
|
public static function checked(array|string $haystack, string $needle, int $type = 0): ?string
|
||||||
{
|
{
|
||||||
if (is_array($haystack)) {
|
if (is_array($haystack) && in_array($needle, $haystack)) {
|
||||||
if (in_array($needle, $haystack)) {
|
return $type ? 'checked' : 'selected';
|
||||||
return $type ? 'checked' : 'selected';
|
} elseif (!is_array($haystack) && $haystack == $needle) {
|
||||||
}
|
return $type ? 'checked' : 'selected';
|
||||||
} else {
|
|
||||||
if ($haystack == $needle) {
|
|
||||||
return $type ? 'checked' : 'selected';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -56,6 +56,180 @@ class Math
|
|||||||
return (float)$number;
|
return (float)$number;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* calc cube root
|
||||||
|
*
|
||||||
|
* @param float $number Number to cubic root
|
||||||
|
* @return float Calculated value
|
||||||
|
*/
|
||||||
|
public static function cbrt(float|int $number): float
|
||||||
|
{
|
||||||
|
return pow((float)$number, 1.0 / 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* use PHP_FLOAT_EPSILON to compare if two float numbers are matching
|
||||||
|
*
|
||||||
|
* @param float $x
|
||||||
|
* @param float $y
|
||||||
|
* @param float $epsilon [default=PHP_FLOAT_EPSILON]
|
||||||
|
* @return bool True equal
|
||||||
|
*/
|
||||||
|
public static function equalWithEpsilon(float $x, float $y, float $epsilon = PHP_FLOAT_EPSILON): bool
|
||||||
|
{
|
||||||
|
if (abs($x - $y) < $epsilon) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compare two value base on direction given
|
||||||
|
* The default delta is PHP_FLOAT_EPSILON
|
||||||
|
*
|
||||||
|
* @param float $value
|
||||||
|
* @param string $compare
|
||||||
|
* @param float $limit
|
||||||
|
* @param float $epsilon [default=PHP_FLOAT_EPSILON]
|
||||||
|
* @return bool True on smaller/large or equal
|
||||||
|
*/
|
||||||
|
public static function compareWithEpsilon(
|
||||||
|
float $value,
|
||||||
|
string $compare,
|
||||||
|
float $limit,
|
||||||
|
float $epsilon = PHP_FLOAT_EPSILON
|
||||||
|
): bool {
|
||||||
|
switch ($compare) {
|
||||||
|
case '<':
|
||||||
|
if ($value < ($limit - $epsilon)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '<=':
|
||||||
|
if ($value <= ($limit - $epsilon)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '==':
|
||||||
|
return self::equalWithEpsilon($value, $limit, $epsilon);
|
||||||
|
case '>':
|
||||||
|
if ($value > ($limit + $epsilon)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '>=':
|
||||||
|
if ($value >= ($limit + $epsilon)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This function is directly inspired by the multiplyMatrices() function in color.js
|
||||||
|
* form Lea Verou and Chris Lilley.
|
||||||
|
* (see https://github.com/LeaVerou/color.js/blob/main/src/multiply-matrices.js)
|
||||||
|
* From:
|
||||||
|
* https://github.com/matthieumastadenis/couleur/blob/3842cf51c9517e77afaa0a36ec78643a0c258e0b/src/utils/utils.php#L507
|
||||||
|
*
|
||||||
|
* It returns an array which is the product of the two number matrices passed as parameters.
|
||||||
|
*
|
||||||
|
* NOTE:
|
||||||
|
* if the right side (B matrix) has a missing row, this row will be fillwed with 0 instead of
|
||||||
|
* throwing an error:
|
||||||
|
* A:
|
||||||
|
* [
|
||||||
|
* [1, 2, 3],
|
||||||
|
* [4, 5, 6],
|
||||||
|
* ]
|
||||||
|
* B:
|
||||||
|
* [
|
||||||
|
* [7, 8, 9],
|
||||||
|
* [10, 11, 12],
|
||||||
|
* ]
|
||||||
|
* The B will get a third row with [0, 0, 0] added to make the multiplication work as it will be
|
||||||
|
* rewritten as
|
||||||
|
* B-rewrite:
|
||||||
|
* [
|
||||||
|
* [7, 10, 0],
|
||||||
|
* [8, 11, 12],
|
||||||
|
* [0, 0, 0] <- automatically added
|
||||||
|
* ]
|
||||||
|
*
|
||||||
|
* The same is done for unbalanced entries, they are filled with 0
|
||||||
|
*
|
||||||
|
* @param array<float|int|array<int|float>> $a m x n matrice
|
||||||
|
* @param array<float|int|array<int|float>> $b n x p matrice
|
||||||
|
*
|
||||||
|
* @return array<float|int|array<int|float>> m x p product
|
||||||
|
*/
|
||||||
|
public static function multiplyMatrices(array $a, array $b): array
|
||||||
|
{
|
||||||
|
$m = count($a);
|
||||||
|
|
||||||
|
if (!is_array($a[0] ?? null)) {
|
||||||
|
// $a is vector, convert to [[a, b, c, ...]]
|
||||||
|
$a = [$a];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!is_array($b[0])) {
|
||||||
|
// $b is vector, convert to [[a], [b], [c], ...]]
|
||||||
|
$b = array_map(
|
||||||
|
callback: fn ($v) => [ $v ],
|
||||||
|
array: $b,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$p = count($b[0]);
|
||||||
|
|
||||||
|
// transpose $b:
|
||||||
|
// so that we can multiply row by row
|
||||||
|
$bCols = array_map(
|
||||||
|
callback: fn ($k) => array_map(
|
||||||
|
(fn ($i) => is_array($i) ? $i[$k] ?? 0 : 0),
|
||||||
|
$b,
|
||||||
|
),
|
||||||
|
array: array_keys($b[0]),
|
||||||
|
);
|
||||||
|
|
||||||
|
$product = array_map(
|
||||||
|
callback: fn ($row) => array_map(
|
||||||
|
callback: fn ($col) => is_array($row) ?
|
||||||
|
array_reduce(
|
||||||
|
array: $row,
|
||||||
|
callback: fn ($a, $v, $i = null) => $a + $v * (
|
||||||
|
// if last entry missing for full copy add a 0 to it
|
||||||
|
$col[$i ?? array_search($v, $row, true)] ?? 0 /** @phpstan-ignore-line */
|
||||||
|
),
|
||||||
|
initial: 0,
|
||||||
|
) :
|
||||||
|
array_reduce(
|
||||||
|
array: $col,
|
||||||
|
callback: fn ($a, $v) => $a + $v * $row,
|
||||||
|
initial: 0,
|
||||||
|
),
|
||||||
|
array: $bCols,
|
||||||
|
),
|
||||||
|
array: $a,
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($m === 1) {
|
||||||
|
// Avoid [[a, b, c, ...]]:
|
||||||
|
return $product[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($p === 1) {
|
||||||
|
// Avoid [[a], [b], [c], ...]]:
|
||||||
|
return array_map(
|
||||||
|
callback: fn ($v) => $v[0] ?? 0,
|
||||||
|
array: $product,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $product;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ class MimeEncode
|
|||||||
/**
|
/**
|
||||||
* wrapper function for mb mime convert
|
* wrapper function for mb mime convert
|
||||||
* for correct conversion with long strings
|
* for correct conversion with long strings
|
||||||
|
* NOTE: This is only a wrapper for mb_encode_mimeheader to stay compatible
|
||||||
*
|
*
|
||||||
* @param string $string string to encode
|
* @param string $string string to encode
|
||||||
* @param string $encoding target encoding
|
* @param string $encoding target encoding
|
||||||
@@ -29,38 +30,9 @@ class MimeEncode
|
|||||||
$current_internal_encoding = mb_internal_encoding();
|
$current_internal_encoding = mb_internal_encoding();
|
||||||
// set internal encoding, so the mimeheader encode works correctly
|
// set internal encoding, so the mimeheader encode works correctly
|
||||||
mb_internal_encoding($encoding);
|
mb_internal_encoding($encoding);
|
||||||
// if a subject, make a work around for the broken mb_mimencode
|
// use the internal convert to mime header
|
||||||
$pos = 0;
|
// it works from PHP 8.2 on
|
||||||
// after 36 single bytes characters,
|
$string = mb_encode_mimeheader($string, $encoding, 'B', $line_break);
|
||||||
// if then comes MB, it is broken
|
|
||||||
// has to 2 x 36 < 74 so the mb_encode_mimeheader
|
|
||||||
// 74 hardcoded split does not get triggered
|
|
||||||
$split = 36;
|
|
||||||
$_string = '';
|
|
||||||
while ($pos < mb_strlen($string, $encoding)) {
|
|
||||||
$output = mb_strimwidth($string, $pos, $split, "", $encoding);
|
|
||||||
$pos += mb_strlen($output, $encoding);
|
|
||||||
// if the strinlen is 0 here, get out of the loop
|
|
||||||
if (!mb_strlen($output, $encoding)) {
|
|
||||||
$pos += mb_strlen($string, $encoding);
|
|
||||||
}
|
|
||||||
$_string_encoded = mb_encode_mimeheader($output, $encoding);
|
|
||||||
// only make linebreaks if we have mime encoded code inside
|
|
||||||
// the space only belongs in the second line
|
|
||||||
if ($_string && preg_match("/^=\?/", $_string_encoded)) {
|
|
||||||
$_string .= $line_break . " ";
|
|
||||||
} elseif (
|
|
||||||
// hack for plain text with space at the end
|
|
||||||
mb_strlen($output, $encoding) == $split &&
|
|
||||||
mb_substr($output, -1, 1, $encoding) == " "
|
|
||||||
) {
|
|
||||||
// if output ends with space, add one more
|
|
||||||
$_string_encoded .= " ";
|
|
||||||
}
|
|
||||||
$_string .= $_string_encoded;
|
|
||||||
}
|
|
||||||
// strip out any spaces BEFORE a line break
|
|
||||||
$string = str_replace(" " . $line_break, $line_break, $_string);
|
|
||||||
// before we end, reset internal encoding
|
// before we end, reset internal encoding
|
||||||
mb_internal_encoding($current_internal_encoding);
|
mb_internal_encoding($current_internal_encoding);
|
||||||
// return mime encoded string
|
// return mime encoded string
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ class SetVarTypeNull extends Extends\SetVarTypeMain
|
|||||||
* @param string|null $default Default override value
|
* @param string|null $default Default override value
|
||||||
* @return string|null Input value as string or default as string/null
|
* @return string|null Input value as string or default as string/null
|
||||||
*/
|
*/
|
||||||
public static function makeStr(mixed $val, string $default = null): ?string
|
public static function makeStr(mixed $val, ?string $default = null): ?string
|
||||||
{
|
{
|
||||||
return SetVarTypeMain::makeStrMain($val, $default, true);
|
return SetVarTypeMain::makeStrMain($val, $default, true);
|
||||||
}
|
}
|
||||||
@@ -60,7 +60,7 @@ class SetVarTypeNull extends Extends\SetVarTypeMain
|
|||||||
* @param int|null $default Default override value
|
* @param int|null $default Default override value
|
||||||
* @return int|null Input value as int or default as int/null
|
* @return int|null Input value as int or default as int/null
|
||||||
*/
|
*/
|
||||||
public static function makeInt(mixed $val, int $default = null): ?int
|
public static function makeInt(mixed $val, ?int $default = null): ?int
|
||||||
{
|
{
|
||||||
return SetVarTypeMain::makeIntMain($val, $default, true);
|
return SetVarTypeMain::makeIntMain($val, $default, true);
|
||||||
}
|
}
|
||||||
@@ -84,7 +84,7 @@ class SetVarTypeNull extends Extends\SetVarTypeMain
|
|||||||
* @param float|null $default Default override value
|
* @param float|null $default Default override value
|
||||||
* @return float|null Input value as float or default as float/null
|
* @return float|null Input value as float or default as float/null
|
||||||
*/
|
*/
|
||||||
public static function makeFloat(mixed $val, float $default = null): ?float
|
public static function makeFloat(mixed $val, ?float $default = null): ?float
|
||||||
{
|
{
|
||||||
return SetVarTypeMain::makeFloatMain($val, $default, true);
|
return SetVarTypeMain::makeFloatMain($val, $default, true);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -118,6 +118,34 @@ class Strings
|
|||||||
return $value;
|
return $value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Strip any duplicated slahes from a path
|
||||||
|
* eg: //foo///bar/foo.inc -> /foo/bar/foo.inc
|
||||||
|
*
|
||||||
|
* @param string $path Path to strip slashes from
|
||||||
|
* @return string Clean path, on error returns original path
|
||||||
|
*/
|
||||||
|
public static function stripMultiplePathSlashes(string $path): string
|
||||||
|
{
|
||||||
|
return preg_replace(
|
||||||
|
'#/+#',
|
||||||
|
'/',
|
||||||
|
$path
|
||||||
|
) ?? $path;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove UTF8 BOM Byte string from line
|
||||||
|
* Note: this is often found in CSV files exported from Excel at the first row, first element
|
||||||
|
*
|
||||||
|
* @param string $text
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public static function stripUTF8BomBytes(string $text): string
|
||||||
|
{
|
||||||
|
return trim($text, pack('H*', 'EFBBBF'));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -15,19 +15,27 @@ namespace CoreLibs\Create;
|
|||||||
|
|
||||||
class Session
|
class Session
|
||||||
{
|
{
|
||||||
|
/** @var string current session name */
|
||||||
|
private string $session_name = '';
|
||||||
|
/** @var string current session id */
|
||||||
|
private string $session_id = '';
|
||||||
|
/** @var bool flag auto write close */
|
||||||
|
private bool $auto_write_close = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* init a session, if array is empty or array does not have session_name set
|
* init a session, if array is empty or array does not have session_name set
|
||||||
* then no auto init is run
|
* then no auto init is run
|
||||||
*
|
*
|
||||||
* @param string $session_name if set and not empty, will start session
|
* @param string $session_name if set and not empty, will start session
|
||||||
*/
|
*/
|
||||||
public function __construct(string $session_name = '')
|
public function __construct(string $session_name, bool $auto_write_close = false)
|
||||||
{
|
{
|
||||||
if (!empty($session_name)) {
|
$this->initSession($session_name);
|
||||||
$this->startSession($session_name);
|
$this->auto_write_close = $auto_write_close;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: private methods
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Start session
|
* Start session
|
||||||
* startSession should be called for complete check
|
* startSession should be called for complete check
|
||||||
@@ -36,36 +44,32 @@ class Session
|
|||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
protected function startSessionCall(): void
|
private function startSessionCall(): void
|
||||||
{
|
{
|
||||||
session_start();
|
session_start();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* check if we are in CLI, we set this, so we can mock this
|
* get current set session id or false if none started
|
||||||
* Not this is just a wrapper for the static System::checkCLI call
|
|
||||||
*
|
*
|
||||||
* @return bool True if we are in a CLI enviroment, or false for everything else
|
* @return string|false
|
||||||
*/
|
*/
|
||||||
public function checkCliStatus(): bool
|
public function getSessionIdCall(): string|false
|
||||||
{
|
{
|
||||||
return \CoreLibs\Get\System::checkCLI();
|
return session_id();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set session name call. If not valid session name, will return false
|
* automatically closes a session if the auto write close flag is set
|
||||||
*
|
*
|
||||||
* @param string $session_name A valid string for session name
|
* @return bool
|
||||||
* @return bool True if session name is valid,
|
|
||||||
* False if not
|
|
||||||
*/
|
*/
|
||||||
public function setSessionName(string $session_name): bool
|
private function closeSessionCall(): bool
|
||||||
{
|
{
|
||||||
if (!$this->checkValidSessionName($session_name)) {
|
if ($this->auto_write_close) {
|
||||||
return false;
|
return $this->writeClose();
|
||||||
}
|
}
|
||||||
session_name($session_name);
|
return false;
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -94,15 +98,34 @@ class Session
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* start session with given session name if set
|
* validate _SESSION key, must be valid variable
|
||||||
|
*
|
||||||
|
* @param int|float|string $key
|
||||||
|
* @return true
|
||||||
|
*/
|
||||||
|
private function checkValidSessionEntryKey(int|float|string $key): true
|
||||||
|
{
|
||||||
|
if (!is_string($key) || is_numeric($key)) {
|
||||||
|
throw new \UnexpectedValueException(
|
||||||
|
'[SESSION] Given key for _SESSION is not a valid value for a varaible: ' . $key,
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: init session (on class start)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* stinitart session with given session name if set
|
||||||
* aborts on command line or if sessions are not enabled
|
* aborts on command line or if sessions are not enabled
|
||||||
* also aborts if session cannot be started
|
* also aborts if session cannot be started
|
||||||
* On sucess returns the session id
|
* On sucess returns the session id
|
||||||
*
|
*
|
||||||
* @param string|null $session_name
|
* @param string $session_name
|
||||||
* @return string|bool
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function startSession(?string $session_name = null): string|bool
|
private function initSession(string $session_name): void
|
||||||
{
|
{
|
||||||
// we can't start sessions on command line
|
// we can't start sessions on command line
|
||||||
if ($this->checkCliStatus()) {
|
if ($this->checkCliStatus()) {
|
||||||
@@ -115,39 +138,85 @@ class Session
|
|||||||
// session_status
|
// session_status
|
||||||
// initial the session if there is no session running already
|
// initial the session if there is no session running already
|
||||||
if (!$this->checkActiveSession()) {
|
if (!$this->checkActiveSession()) {
|
||||||
// if session name is emtpy, check if there is a global set
|
// invalid session name, abort
|
||||||
// this is a deprecated fallback
|
if (!$this->checkValidSessionName($session_name)) {
|
||||||
$session_name = $session_name ?? $GLOBALS['SET_SESSION_NAME'] ?? '';
|
throw new \UnexpectedValueException('[SESSION] Invalid session name: ' . $this->session_name, 3);
|
||||||
// DEPRECTED: constant SET_SESSION_NAME is no longer used
|
|
||||||
// if set, set special session name
|
|
||||||
if (!empty($session_name)) {
|
|
||||||
// invalid session name, abort
|
|
||||||
if (!$this->checkValidSessionName($session_name)) {
|
|
||||||
throw new \UnexpectedValueException('[SESSION] Invalid session name: ' . $session_name, 3);
|
|
||||||
}
|
|
||||||
$this->setSessionName($session_name);
|
|
||||||
}
|
}
|
||||||
|
// set session name
|
||||||
|
$this->session_name = $session_name;
|
||||||
|
session_name($this->session_name);
|
||||||
// start session
|
// start session
|
||||||
$this->startSessionCall();
|
$this->startSessionCall();
|
||||||
|
// if we faild to start the session
|
||||||
|
if (!$this->checkActiveSession()) {
|
||||||
|
throw new \RuntimeException('[SESSION] Failed to activate session', 5);
|
||||||
|
}
|
||||||
|
} elseif ($session_name != $this->getSessionName()) {
|
||||||
|
throw new \UnexpectedValueException(
|
||||||
|
'[SESSION] Another session exists with a different name: ' . $this->getSessionName(),
|
||||||
|
4
|
||||||
|
);
|
||||||
}
|
}
|
||||||
// if we still have no active session
|
// check session id
|
||||||
|
if (false === ($session_id = $this->getSessionIdCall())) {
|
||||||
|
throw new \UnexpectedValueException('[SESSION] getSessionId did not return a session id', 6);
|
||||||
|
}
|
||||||
|
// set session id
|
||||||
|
$this->session_id = $session_id;
|
||||||
|
// if flagged auto close, write close session
|
||||||
|
if ($this->auto_write_close) {
|
||||||
|
$this->writeClose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: public set/get status
|
||||||
|
|
||||||
|
/**
|
||||||
|
* start session, will only run after initSession
|
||||||
|
*
|
||||||
|
* @return bool True if started, False if alrady running
|
||||||
|
*/
|
||||||
|
public function restartSession(): bool
|
||||||
|
{
|
||||||
if (!$this->checkActiveSession()) {
|
if (!$this->checkActiveSession()) {
|
||||||
throw new \RuntimeException('[SESSION] Failed to activate session', 4);
|
if (empty($this->session_name)) {
|
||||||
|
throw new \RuntimeException('[SESSION] Cannot restart session without a session name', 1);
|
||||||
|
}
|
||||||
|
$this->startSessionCall();
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
if (false === ($session_id = $this->getSessionId())) {
|
return false;
|
||||||
throw new \UnexpectedValueException('[SESSION] getSessionId did not return a session id', 5);
|
|
||||||
}
|
|
||||||
return $session_id;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* get current set session id or false if none started
|
* current set session id
|
||||||
*
|
*
|
||||||
* @return string|bool
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function getSessionId(): string|bool
|
public function getSessionId(): string
|
||||||
{
|
{
|
||||||
return session_id();
|
return $this->session_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set the auto write close flag
|
||||||
|
*
|
||||||
|
* @param bool $flag
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function setAutoWriteClose(bool $flag): void
|
||||||
|
{
|
||||||
|
$this->auto_write_close = $flag;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* return the auto write close flag
|
||||||
|
*
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function checkAutoWriteClose(): bool
|
||||||
|
{
|
||||||
|
return $this->auto_write_close;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -175,6 +244,34 @@ class Session
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* check if we are in CLI, we set this, so we can mock this
|
||||||
|
* Not this is just a wrapper for the static System::checkCLI call
|
||||||
|
*
|
||||||
|
* @return bool True if we are in a CLI enviroment, or false for everything else
|
||||||
|
*/
|
||||||
|
public function checkCliStatus(): bool
|
||||||
|
{
|
||||||
|
return \CoreLibs\Get\System::checkCLI();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get session status
|
||||||
|
* PHP_SESSION_DISABLED if sessions are disabled.
|
||||||
|
* PHP_SESSION_NONE if sessions are enabled, but none exists.
|
||||||
|
* PHP_SESSION_ACTIVE if sessions are enabled, and one exists.
|
||||||
|
*
|
||||||
|
* https://www.php.net/manual/en/function.session-status.php
|
||||||
|
*
|
||||||
|
* @return int See possible return int values above
|
||||||
|
*/
|
||||||
|
public function getSessionStatus(): int
|
||||||
|
{
|
||||||
|
return session_status();
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: write close session
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* unlock the session file, so concurrent AJAX requests can be done
|
* unlock the session file, so concurrent AJAX requests can be done
|
||||||
* NOTE: after this has been called, no changes in _SESSION will be stored
|
* NOTE: after this has been called, no changes in _SESSION will be stored
|
||||||
@@ -188,17 +285,24 @@ class Session
|
|||||||
return session_write_close();
|
return session_write_close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: session close and clean up
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Proper destroy a session
|
* Proper destroy a session
|
||||||
* - unset the _SESSION array
|
* - unset the _SESSION array
|
||||||
* - unset cookie if cookie on and we have not strict mode
|
* - unset cookie if cookie on and we have not strict mode
|
||||||
|
* - unset session_name and session_id internal vars
|
||||||
* - destroy session
|
* - destroy session
|
||||||
*
|
*
|
||||||
* @return bool
|
* @return bool True on successful session destroy
|
||||||
*/
|
*/
|
||||||
public function sessionDestroy(): bool
|
public function sessionDestroy(): bool
|
||||||
{
|
{
|
||||||
$_SESSION = [];
|
// abort to false if not unsetable
|
||||||
|
if (!session_unset()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
$this->clear();
|
||||||
if (
|
if (
|
||||||
ini_get('session.use_cookies') &&
|
ini_get('session.use_cookies') &&
|
||||||
!ini_get('session.use_strict_mode')
|
!ini_get('session.use_strict_mode')
|
||||||
@@ -218,68 +322,92 @@ class Session
|
|||||||
$params['httponly']
|
$params['httponly']
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
// unset internal vars
|
||||||
|
$this->session_name = '';
|
||||||
|
$this->session_id = '';
|
||||||
return session_destroy();
|
return session_destroy();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
// MARK: _SESSION set/unset methods
|
||||||
* get session status
|
|
||||||
* PHP_SESSION_DISABLED if sessions are disabled.
|
|
||||||
* PHP_SESSION_NONE if sessions are enabled, but none exists.
|
|
||||||
* PHP_SESSION_ACTIVE if sessions are enabled, and one exists.
|
|
||||||
*
|
|
||||||
* https://www.php.net/manual/en/function.session-status.php
|
|
||||||
*
|
|
||||||
* @return int See possible return int values above
|
|
||||||
*/
|
|
||||||
public function getSessionStatus(): int
|
|
||||||
{
|
|
||||||
return session_status();
|
|
||||||
}
|
|
||||||
|
|
||||||
// _SESSION set/unset methods
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* unset all _SESSION entries
|
* unset all _SESSION entries
|
||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function unsetAllS(): void
|
public function clear(): void
|
||||||
{
|
{
|
||||||
foreach (array_keys($_SESSION ?? []) as $name) {
|
$this->restartSession();
|
||||||
unset($_SESSION[$name]);
|
if (!session_unset()) {
|
||||||
|
throw new \RuntimeException('[SESSION] Cannot unset session vars', 1);
|
||||||
}
|
}
|
||||||
|
if (!empty($_SESSION)) {
|
||||||
|
$_SESSION = [];
|
||||||
|
}
|
||||||
|
$this->closeSessionCall();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* set _SESSION entry 'name' with any value
|
* set _SESSION entry 'name' with any value
|
||||||
*
|
*
|
||||||
* @param string|int $name array name in _SESSION
|
* @param string $name array name in _SESSION
|
||||||
* @param mixed $value value to set (can be anything)
|
* @param mixed $value value to set (can be anything)
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function setS(string|int $name, mixed $value): void
|
public function set(string $name, mixed $value): void
|
||||||
{
|
{
|
||||||
|
$this->checkValidSessionEntryKey($name);
|
||||||
|
$this->restartSession();
|
||||||
$_SESSION[$name] = $value;
|
$_SESSION[$name] = $value;
|
||||||
|
$this->closeSessionCall();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set many session entries in one set
|
||||||
|
*
|
||||||
|
* @param array<string,mixed> $set key is the key in the _SESSION, value is any data to set
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function setMany(array $set): void
|
||||||
|
{
|
||||||
|
$this->restartSession();
|
||||||
|
// skip any that are not valid
|
||||||
|
foreach ($set as $key => $value) {
|
||||||
|
$this->checkValidSessionEntryKey($key);
|
||||||
|
$_SESSION[$key] = $value;
|
||||||
|
}
|
||||||
|
$this->closeSessionCall();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* get _SESSION 'name' entry or empty string if not set
|
* get _SESSION 'name' entry or empty string if not set
|
||||||
*
|
*
|
||||||
* @param string|int $name value key to get from _SESSION
|
* @param string $name value key to get from _SESSION
|
||||||
* @return mixed value stored in _SESSION
|
* @return mixed value stored in _SESSION, if not found set to null
|
||||||
*/
|
*/
|
||||||
public function getS(string|int $name): mixed
|
public function get(string $name): mixed
|
||||||
{
|
{
|
||||||
return $_SESSION[$name] ?? '';
|
return $_SESSION[$name] ?? null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get multiple session entries
|
||||||
|
*
|
||||||
|
* @param array<string> $set
|
||||||
|
* @return array<string,mixed>
|
||||||
|
*/
|
||||||
|
public function getMany(array $set): array
|
||||||
|
{
|
||||||
|
return array_intersect_key($_SESSION, array_flip($set));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a name is set in the _SESSION array
|
* Check if a name is set in the _SESSION array
|
||||||
*
|
*
|
||||||
* @param string|int $name Name to check for
|
* @param string $name Name to check for
|
||||||
* @return bool True for set, False fornot set
|
* @return bool True for set, False fornot set
|
||||||
*/
|
*/
|
||||||
public function issetS(string|int $name): bool
|
public function isset(string $name): bool
|
||||||
{
|
{
|
||||||
return isset($_SESSION[$name]);
|
return isset($_SESSION[$name]);
|
||||||
}
|
}
|
||||||
@@ -287,67 +415,35 @@ class Session
|
|||||||
/**
|
/**
|
||||||
* unset one _SESSION entry 'name' if exists
|
* unset one _SESSION entry 'name' if exists
|
||||||
*
|
*
|
||||||
* @param string|int $name _SESSION key name to remove
|
* @param string $name _SESSION key name to remove
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function unsetS(string|int $name): void
|
public function unset(string $name): void
|
||||||
{
|
{
|
||||||
if (isset($_SESSION[$name])) {
|
if (!isset($_SESSION[$name])) {
|
||||||
unset($_SESSION[$name]);
|
return;
|
||||||
}
|
}
|
||||||
|
$this->restartSession();
|
||||||
|
unset($_SESSION[$name]);
|
||||||
|
$this->closeSessionCall();
|
||||||
}
|
}
|
||||||
|
|
||||||
// set/get below
|
|
||||||
// ->var = value;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* reset many session entry
|
||||||
*
|
*
|
||||||
* @param string|int $name
|
* @param array<string> $set list of session keys to reset
|
||||||
* @param mixed $value
|
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function __set(string|int $name, mixed $value): void
|
public function unsetMany(array $set): void
|
||||||
{
|
{
|
||||||
$_SESSION[$name] = $value;
|
$this->restartSession();
|
||||||
}
|
foreach ($set as $key) {
|
||||||
|
if (!isset($_SESSION[$key])) {
|
||||||
/**
|
continue;
|
||||||
* Undocumented function
|
}
|
||||||
*
|
unset($_SESSION[$key]);
|
||||||
* @param string|int $name
|
|
||||||
* @return mixed If name is not found, it will return null
|
|
||||||
*/
|
|
||||||
public function __get(string|int $name): mixed
|
|
||||||
{
|
|
||||||
if (isset($_SESSION[$name])) {
|
|
||||||
return $_SESSION[$name];
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @param string|int $name
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function __isset(string|int $name): bool
|
|
||||||
{
|
|
||||||
return isset($_SESSION[$name]);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @param string|int $name
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function __unset(string|int $name): void
|
|
||||||
{
|
|
||||||
if (isset($_SESSION[$name])) {
|
|
||||||
unset($_SESSION[$name]);
|
|
||||||
}
|
}
|
||||||
|
$this->closeSessionCall();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ class Uids
|
|||||||
$uniqid_length++;
|
$uniqid_length++;
|
||||||
}
|
}
|
||||||
/** @var int<1,max> make sure that internal this is correct */
|
/** @var int<1,max> make sure that internal this is correct */
|
||||||
$random_bytes_length = ($uniqid_length - ($uniqid_length % 2)) / 2;
|
$random_bytes_length = (int)(($uniqid_length - ($uniqid_length % 2)) / 2);
|
||||||
$uniqid = bin2hex(random_bytes($random_bytes_length));
|
$uniqid = bin2hex(random_bytes($random_bytes_length));
|
||||||
// if not forced shorten return next lower length
|
// if not forced shorten return next lower length
|
||||||
if (!$force_length) {
|
if (!$force_length) {
|
||||||
@@ -56,26 +56,6 @@ class Uids
|
|||||||
*/
|
*/
|
||||||
public static function uuidv4(): string
|
public static function uuidv4(): string
|
||||||
{
|
{
|
||||||
/* return sprintf(
|
|
||||||
'%04x%04x-%04x-%04x-%04x-%04x%04x%04x',
|
|
||||||
// 32 bits for "time_low"
|
|
||||||
mt_rand(0, 0xffff),
|
|
||||||
mt_rand(0, 0xffff),
|
|
||||||
// 16 bits for "time_mid"
|
|
||||||
mt_rand(0, 0xffff),
|
|
||||||
// 16 bits for "time_hi_and_version",
|
|
||||||
// four most significant bits holds version number 4
|
|
||||||
mt_rand(0, 0x0fff) | 0x4000,
|
|
||||||
// 16 bits, 8 bits for "clk_seq_hi_res",
|
|
||||||
// 8 bits for "clk_seq_low",
|
|
||||||
// two most significant bits holds zero and one for variant DCE1.1
|
|
||||||
mt_rand(0, 0x3fff) | 0x8000,
|
|
||||||
// 48 bits for "node"
|
|
||||||
mt_rand(0, 0xffff),
|
|
||||||
mt_rand(0, 0xffff),
|
|
||||||
mt_rand(0, 0xffff)
|
|
||||||
); */
|
|
||||||
|
|
||||||
$data = random_bytes(16);
|
$data = random_bytes(16);
|
||||||
assert(strlen($data) == 16);
|
assert(strlen($data) == 16);
|
||||||
|
|
||||||
@@ -93,6 +73,20 @@ class Uids
|
|||||||
return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
|
return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* regex validate uuid v4
|
||||||
|
*
|
||||||
|
* @param string $uuidv4
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public static function validateUuuidv4(string $uuidv4): bool
|
||||||
|
{
|
||||||
|
if (!preg_match("/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/", $uuidv4)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* creates a uniq id based on lengths
|
* creates a uniq id based on lengths
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -55,7 +55,7 @@ class ArrayIO extends \CoreLibs\DB\IO
|
|||||||
* primary key name automatically (from array)
|
* primary key name automatically (from array)
|
||||||
*
|
*
|
||||||
* phpcs:ignore
|
* phpcs:ignore
|
||||||
* @param array{db_name:string,db_user:string,db_pass:string,db_host:string,db_port:int,db_schema:string,db_encoding:string,db_type:string,db_ssl:string,db_convert_type?:string[]} $db_config db connection config
|
* @param array{db_name:string,db_user:string,db_pass:string,db_host:string,db_port:int,db_schema:string,db_encoding:string,db_type:string,db_ssl:string,db_convert_type?:string[],db_convert_placeholder?:bool,db_convert_placeholder_target?:string,db_debug_replace_placeholder?:bool} $db_config db connection config
|
||||||
* @param array<mixed> $table_array table array config
|
* @param array<mixed> $table_array table array config
|
||||||
* @param string $table_name table name string
|
* @param string $table_name table name string
|
||||||
* @param \CoreLibs\Logging\Logging $log Logging class
|
* @param \CoreLibs\Logging\Logging $log Logging class
|
||||||
@@ -236,7 +236,7 @@ class ArrayIO extends \CoreLibs\DB\IO
|
|||||||
*/
|
*/
|
||||||
public function getPkId(): int|string|null
|
public function getPkId(): int|string|null
|
||||||
{
|
{
|
||||||
return $this->pk_id;
|
return $this->pk_id ?? null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -374,7 +374,7 @@ class ArrayIO extends \CoreLibs\DB\IO
|
|||||||
public function dbDelete(array $table_array = [], bool $acl_limit = false): array
|
public function dbDelete(array $table_array = [], bool $acl_limit = false): array
|
||||||
{
|
{
|
||||||
// is array and has values, override set and set new
|
// is array and has values, override set and set new
|
||||||
if (is_array($table_array) && count($table_array)) {
|
if (count($table_array)) {
|
||||||
$this->table_array = $table_array;
|
$this->table_array = $table_array;
|
||||||
}
|
}
|
||||||
if (!$this->dbCheckPkSet()) {
|
if (!$this->dbCheckPkSet()) {
|
||||||
@@ -440,7 +440,7 @@ class ArrayIO extends \CoreLibs\DB\IO
|
|||||||
public function dbRead(bool $edit = false, array $table_array = []): array
|
public function dbRead(bool $edit = false, array $table_array = []): array
|
||||||
{
|
{
|
||||||
// if array give, overrules internal array
|
// if array give, overrules internal array
|
||||||
if (is_array($table_array) && count($table_array)) {
|
if (count($table_array)) {
|
||||||
$this->table_array = $table_array;
|
$this->table_array = $table_array;
|
||||||
}
|
}
|
||||||
if (!$this->dbCheckPkSet()) {
|
if (!$this->dbCheckPkSet()) {
|
||||||
|
|||||||
922
src/DB/IO.php
922
src/DB/IO.php
File diff suppressed because it is too large
Load Diff
@@ -285,6 +285,22 @@ interface SqlFunctions
|
|||||||
*/
|
*/
|
||||||
public function __dbConnectionBusySocketWait(int $timeout_seconds = 3): bool;
|
public function __dbConnectionBusySocketWait(int $timeout_seconds = 3): bool;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @param string $parameter
|
||||||
|
* @param bool $strip
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function __dbVersionInfo(string $parameter, bool $strip = true): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array<mixed>
|
||||||
|
*/
|
||||||
|
public function __dbVersionInfoParameterList(): array;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -292,6 +308,13 @@ interface SqlFunctions
|
|||||||
*/
|
*/
|
||||||
public function __dbVersion(): string;
|
public function __dbVersion(): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
public function __dbVersionNumeric(): int;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -306,6 +329,14 @@ interface SqlFunctions
|
|||||||
?int &$end = null
|
?int &$end = null
|
||||||
): ?array;
|
): ?array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @param string $parameter
|
||||||
|
* @return string|bool
|
||||||
|
*/
|
||||||
|
public function __dbParameter(string $parameter): string|bool;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -343,6 +374,14 @@ interface SqlFunctions
|
|||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function __dbGetEncoding(): string;
|
public function __dbGetEncoding(): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @param string $query
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
public function __dbCountQueryParams(string $query): int;
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -51,6 +51,8 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace CoreLibs\DB\SQL;
|
namespace CoreLibs\DB\SQL;
|
||||||
|
|
||||||
|
use CoreLibs\DB\Support\ConvertPlaceholder;
|
||||||
|
|
||||||
// below no ignore is needed if we want to use PgSql interface checks with PHP 8.0
|
// below no ignore is needed if we want to use PgSql interface checks with PHP 8.0
|
||||||
// as main system. Currently all @var sets are written as object
|
// as main system. Currently all @var sets are written as object
|
||||||
/** @#phan-file-suppress PhanUndeclaredTypeProperty,PhanUndeclaredTypeParameter,PhanUndeclaredTypeReturnType */
|
/** @#phan-file-suppress PhanUndeclaredTypeProperty,PhanUndeclaredTypeParameter,PhanUndeclaredTypeReturnType */
|
||||||
@@ -102,7 +104,7 @@ class PgSQL implements Interface\SqlFunctions
|
|||||||
* SELECT foo FROM bar WHERE foobar = $1
|
* SELECT foo FROM bar WHERE foobar = $1
|
||||||
*
|
*
|
||||||
* @param string $query Query string with placeholders $1, ..
|
* @param string $query Query string with placeholders $1, ..
|
||||||
* @param array<mixed> $params Matching parameters for each placerhold
|
* @param array<mixed> $params Matching parameters for each placeholder
|
||||||
* @return \PgSql\Result|false Query result
|
* @return \PgSql\Result|false Query result
|
||||||
*/
|
*/
|
||||||
public function __dbQueryParams(string $query, array $params): \PgSql\Result|false
|
public function __dbQueryParams(string $query, array $params): \PgSql\Result|false
|
||||||
@@ -140,7 +142,7 @@ class PgSQL implements Interface\SqlFunctions
|
|||||||
* sends an async query to the server with params
|
* sends an async query to the server with params
|
||||||
*
|
*
|
||||||
* @param string $query Query string with placeholders $1, ..
|
* @param string $query Query string with placeholders $1, ..
|
||||||
* @param array<mixed> $params Matching parameters for each placerhold
|
* @param array<mixed> $params Matching parameters for each placeholder
|
||||||
* @return bool true/false Query sent successful status
|
* @return bool true/false Query sent successful status
|
||||||
*/
|
*/
|
||||||
public function __dbSendQueryParams(string $query, array $params): bool
|
public function __dbSendQueryParams(string $query, array $params): bool
|
||||||
@@ -405,17 +407,13 @@ class PgSQL implements Interface\SqlFunctions
|
|||||||
}
|
}
|
||||||
// no PK name given at all
|
// no PK name given at all
|
||||||
if (empty($pk_name)) {
|
if (empty($pk_name)) {
|
||||||
// if name is plurar, make it singular
|
|
||||||
// if (preg_match("/.*s$/i", $table))
|
|
||||||
// $table = substr($table, 0, -1);
|
|
||||||
// set pk_name to "id"
|
// set pk_name to "id"
|
||||||
$pk_name = $table . "_id";
|
$pk_name = $table . "_id";
|
||||||
}
|
}
|
||||||
$seq = ($schema ? $schema . '.' : '') . $table . "_" . $pk_name . "_seq";
|
$q = "SELECT CURRVAL(pg_get_serial_sequence($1, $2)) AS insert_id";
|
||||||
$q = "SELECT CURRVAL('$seq') AS insert_id";
|
|
||||||
// I have to do manually or I overwrite the original insert internal vars ...
|
// I have to do manually or I overwrite the original insert internal vars ...
|
||||||
if ($q = $this->__dbQuery($q)) {
|
if ($cursor = $this->__dbQueryParams($q, [$table, $pk_name])) {
|
||||||
if (is_array($res = $this->__dbFetchArray($q))) {
|
if (is_array($res = $this->__dbFetchArray($cursor))) {
|
||||||
list($id) = $res;
|
list($id) = $res;
|
||||||
} else {
|
} else {
|
||||||
return false;
|
return false;
|
||||||
@@ -449,26 +447,36 @@ class PgSQL implements Interface\SqlFunctions
|
|||||||
$table_prefix = $schema . '.';
|
$table_prefix = $schema . '.';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
$params = [$table_prefix . $table];
|
||||||
|
$replace = ['', ''];
|
||||||
// read from table the PK name
|
// read from table the PK name
|
||||||
// faster primary key get
|
// faster primary key get
|
||||||
$q = "SELECT pg_attribute.attname AS column_name, "
|
$q = <<<SQL
|
||||||
. "format_type(pg_attribute.atttypid, pg_attribute.atttypmod) AS type "
|
SELECT
|
||||||
. "FROM pg_index, pg_class, pg_attribute ";
|
pg_attribute.attname AS column_name,
|
||||||
|
format_type(pg_attribute.atttypid, pg_attribute.atttypmod) AS type
|
||||||
|
FROM pg_index, pg_class, pg_attribute{PG_NAMESPACE}
|
||||||
|
WHERE
|
||||||
|
-- regclass translates the OID to the name
|
||||||
|
pg_class.oid = $1::regclass AND
|
||||||
|
indrelid = pg_class.oid AND
|
||||||
|
pg_attribute.attrelid = pg_class.oid AND
|
||||||
|
pg_attribute.attnum = any(pg_index.indkey) AND
|
||||||
|
indisprimary
|
||||||
|
{NSPNAME}
|
||||||
|
SQL;
|
||||||
if ($schema) {
|
if ($schema) {
|
||||||
$q .= ", pg_namespace ";
|
$params[] = $schema;
|
||||||
|
$replace = [
|
||||||
|
", pg_namespace",
|
||||||
|
"AND pg_class.relnamespace = pg_namespace.oid AND nspname = $2"
|
||||||
|
];
|
||||||
}
|
}
|
||||||
$q .= "WHERE "
|
$cursor = $this->__dbQueryParams(str_replace(
|
||||||
// regclass translates the OID to the name
|
['{PG_NAMESPACE}', '{NSPNAME}'],
|
||||||
. "pg_class.oid = '" . $table_prefix . $table . "'::regclass AND "
|
$replace,
|
||||||
. "indrelid = pg_class.oid AND ";
|
$q
|
||||||
if ($schema) {
|
), $params);
|
||||||
$q .= "nspname = '" . $schema . "' AND "
|
|
||||||
. "pg_class.relnamespace = pg_namespace.oid AND ";
|
|
||||||
}
|
|
||||||
$q .= "pg_attribute.attrelid = pg_class.oid AND "
|
|
||||||
. "pg_attribute.attnum = any(pg_index.indkey) "
|
|
||||||
. "AND indisprimary";
|
|
||||||
$cursor = $this->__dbQuery($q);
|
|
||||||
if ($cursor !== false) {
|
if ($cursor !== false) {
|
||||||
$__db_fetch_array = $this->__dbFetchArray($cursor);
|
$__db_fetch_array = $this->__dbFetchArray($cursor);
|
||||||
if (!is_array($__db_fetch_array)) {
|
if (!is_array($__db_fetch_array)) {
|
||||||
@@ -893,11 +901,13 @@ class PgSQL implements Interface\SqlFunctions
|
|||||||
public function __dbSetSchema(string $db_schema): int
|
public function __dbSetSchema(string $db_schema): int
|
||||||
{
|
{
|
||||||
// check if schema actually exists
|
// check if schema actually exists
|
||||||
$query = "SELECT EXISTS("
|
$query = <<<SQL
|
||||||
. "SELECT 1 FROM information_schema.schemata "
|
SELECT EXISTS (
|
||||||
. "WHERE schema_name = " . $this->__dbEscapeLiteral($db_schema)
|
SELECT 1 FROM information_schema.schemata
|
||||||
. ")";
|
WHERE schema_name = $1
|
||||||
$cursor = $this->__dbQuery($query);
|
)
|
||||||
|
SQL;
|
||||||
|
$cursor = $this->__dbQueryParams($query, [$db_schema]);
|
||||||
// abort if execution fails
|
// abort if execution fails
|
||||||
if ($cursor === false) {
|
if ($cursor === false) {
|
||||||
return 1;
|
return 1;
|
||||||
@@ -966,6 +976,34 @@ class PgSQL implements Interface\SqlFunctions
|
|||||||
{
|
{
|
||||||
return $this->__dbShow('client_encoding');
|
return $this->__dbShow('client_encoding');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Count placeholder queries. $ only
|
||||||
|
*
|
||||||
|
* @param string $query
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
public function __dbCountQueryParams(string $query): int
|
||||||
|
{
|
||||||
|
$matches = [];
|
||||||
|
// regex for params: only stand alone $number allowed
|
||||||
|
// exclude all '' enclosed strings, ignore all numbers [note must start with digit]
|
||||||
|
// can have space/tab/new line
|
||||||
|
// must have <> = , ( [not equal, equal, comma, opening round bracket]
|
||||||
|
// can have space/tab/new line
|
||||||
|
// $ number with 1-9 for first and 0-9 for further digits
|
||||||
|
// Collects also PDO ? and :named, but they are ignored
|
||||||
|
// /s for matching new line in . list
|
||||||
|
// [disabled, we don't used ^ or $] /m for multi line match
|
||||||
|
// Matches in 1:, must be array_filtered to remove empty, count with array_unique
|
||||||
|
// Regex located in the ConvertPlaceholder class
|
||||||
|
preg_match_all(
|
||||||
|
ConvertPlaceholder::REGEX_LOOKUP_PLACEHOLDERS,
|
||||||
|
$query,
|
||||||
|
$matches
|
||||||
|
);
|
||||||
|
return count(array_unique(array_filter($matches[3])));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
338
src/DB/Support/ConvertPlaceholder.php
Normal file
338
src/DB/Support/ConvertPlaceholder.php
Normal file
@@ -0,0 +1,338 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2023/10/10
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Convert placeholders in query from PDO style ? or :named to \PG style $number
|
||||||
|
* pr the other way around
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\DB\Support;
|
||||||
|
|
||||||
|
class ConvertPlaceholder
|
||||||
|
{
|
||||||
|
// NOTE for missing: range */+ are not iplemented in the regex below, but - is for now
|
||||||
|
// NOTE some combinations are allowed, but the query will fail before this
|
||||||
|
/** @var string split regex, entries before $ group */
|
||||||
|
private const PATTERN_QUERY_SPLIT =
|
||||||
|
',|' // for ',' mostly in INSERT
|
||||||
|
. '[(<>=]|' // general set for (, <, >, = in any query with any combination
|
||||||
|
. '(?:[\(,]\s*\-\-[\s\w]*)\r?\n|' // a comment that starts after a ( or ,
|
||||||
|
. '\^@|' // text search for start from text with ^@
|
||||||
|
. '\|\||' // concats two elements
|
||||||
|
. '&&|' // array overlap
|
||||||
|
. '\-\|\-|' // range overlap
|
||||||
|
. '[^-]-{1}|' // single -, used in JSON too
|
||||||
|
. '->|->>|#>|#>>|@>|<@|@@|@\?|\?{1}|\?\||\?&|#-'; //JSON searches, Array searchs, etc
|
||||||
|
/** @var string the main regex including the pattern query split */
|
||||||
|
private const PATTERN_ELEMENT = '(?:\'.*?\')?\s*(?:\?\?|' . self::PATTERN_QUERY_SPLIT . ')\s*';
|
||||||
|
/** @var string parts to ignore in the SQL */
|
||||||
|
private const PATTERN_IGNORE =
|
||||||
|
// digit -> ignore
|
||||||
|
'\d+|'
|
||||||
|
// other string -> ignore
|
||||||
|
. '(?:\'.*?\')|';
|
||||||
|
/** @var string named parameters */
|
||||||
|
private const PATTERN_NAMED = '(:\w+)';
|
||||||
|
/** @var string question mark parameters */
|
||||||
|
private const PATTERN_QUESTION_MARK = '(?:(?:\?\?)?\s*(\?{1}))';
|
||||||
|
/** @var string numbered parameters */
|
||||||
|
private const PATTERN_NUMBERED = '(\$[1-9]{1}(?:[0-9]{1,})?)';
|
||||||
|
// below here are full regex that will be used
|
||||||
|
/** @var string replace regex for named (:...) entries */
|
||||||
|
public const REGEX_REPLACE_NAMED = '/'
|
||||||
|
. '(' . self::PATTERN_ELEMENT . ')'
|
||||||
|
. '('
|
||||||
|
. self::PATTERN_IGNORE
|
||||||
|
. self::PATTERN_NAMED
|
||||||
|
. ')'
|
||||||
|
. '/s';
|
||||||
|
/** @var string replace regex for question mark (?) entries */
|
||||||
|
public const REGEX_REPLACE_QUESTION_MARK = '/'
|
||||||
|
. '(' . self::PATTERN_ELEMENT . ')'
|
||||||
|
. '('
|
||||||
|
. self::PATTERN_IGNORE
|
||||||
|
. self::PATTERN_QUESTION_MARK
|
||||||
|
. ')'
|
||||||
|
. '/s';
|
||||||
|
/** @var string replace regex for numbered ($n) entries */
|
||||||
|
public const REGEX_REPLACE_NUMBERED = '/'
|
||||||
|
. '(' . self::PATTERN_ELEMENT . ')'
|
||||||
|
. '('
|
||||||
|
. self::PATTERN_IGNORE
|
||||||
|
. self::PATTERN_NUMBERED
|
||||||
|
. ')'
|
||||||
|
. '/s';
|
||||||
|
/** @var string the main lookup query for all placeholders */
|
||||||
|
public const REGEX_LOOKUP_PLACEHOLDERS = '/'
|
||||||
|
// prefix string part, must match towards
|
||||||
|
// seperator for ( = , ? - [and json/jsonb in pg doc section 9.15]
|
||||||
|
. self::PATTERN_ELEMENT
|
||||||
|
// match for replace part
|
||||||
|
. '(?:'
|
||||||
|
// ignore parts
|
||||||
|
. self::PATTERN_IGNORE
|
||||||
|
// :name named part (PDO) [1]
|
||||||
|
. self::PATTERN_NAMED . '|'
|
||||||
|
// ? question mark part (PDO) [2]
|
||||||
|
. self::PATTERN_QUESTION_MARK . '|'
|
||||||
|
// $n numbered part (\PG php) [3]
|
||||||
|
. self::PATTERN_NUMBERED
|
||||||
|
// end match
|
||||||
|
. ')'
|
||||||
|
// single line -> add line break to matches in "."
|
||||||
|
. '/s';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert PDO type query with placeholders to \PG style and vica versa
|
||||||
|
* For PDO to: ? and :named
|
||||||
|
* For \PG to: $number
|
||||||
|
*
|
||||||
|
* If the query has a mix of ?, :named or $numbrer the \OutOfRangeException exception
|
||||||
|
* will be thrown
|
||||||
|
*
|
||||||
|
* If the convert_to is either pg or pdo, nothing will be changed
|
||||||
|
*
|
||||||
|
* found has -1 if an error occoured in the preg_match_all call
|
||||||
|
*
|
||||||
|
* @param string $query Query with placeholders to convert
|
||||||
|
* @param ?array<mixed> $params The parameters that are used for the query, and will be updated
|
||||||
|
* @param string $convert_to Either pdo or pg, will be converted to lower case for check
|
||||||
|
* @return array{original:array{query:string,params:array<mixed>,empty_params:bool},type:''|'named'|'numbered'|'question_mark',found:int,matches:array<string>,params_lookup:array<mixed>,query:string,params:array<mixed>}
|
||||||
|
* @throws \OutOfRangeException 200 If mixed placeholder types
|
||||||
|
* @throws \InvalidArgumentException 300 or 301 if wrong convert to with found placeholders
|
||||||
|
*/
|
||||||
|
public static function convertPlaceholderInQuery(
|
||||||
|
string $query,
|
||||||
|
?array $params,
|
||||||
|
string $convert_to = 'pg'
|
||||||
|
): array {
|
||||||
|
$convert_to = strtolower($convert_to);
|
||||||
|
$matches = [];
|
||||||
|
// matches:
|
||||||
|
// 1: :named
|
||||||
|
// 2: ? question mark
|
||||||
|
// 3: $n numbered
|
||||||
|
$found = preg_match_all(self::REGEX_LOOKUP_PLACEHOLDERS, $query, $matches, PREG_UNMATCHED_AS_NULL);
|
||||||
|
// if false or null set to -1
|
||||||
|
// || $found === null
|
||||||
|
if ($found === false) {
|
||||||
|
$found = -1;
|
||||||
|
}
|
||||||
|
/** @var array<string> 1: named */
|
||||||
|
$named_matches = array_filter($matches[1]);
|
||||||
|
/** @var array<string> 2: open ? */
|
||||||
|
$qmark_matches = array_filter($matches[2]);
|
||||||
|
/** @var array<string> 3: $n matches */
|
||||||
|
$numbered_matches = array_filter($matches[3]);
|
||||||
|
// count matches
|
||||||
|
$count_named = count(array_unique($named_matches));
|
||||||
|
$count_qmark = count($qmark_matches);
|
||||||
|
$count_numbered = count(array_unique($numbered_matches));
|
||||||
|
// throw exception if mixed found
|
||||||
|
if (
|
||||||
|
($count_named && $count_qmark) ||
|
||||||
|
($count_named && $count_numbered) ||
|
||||||
|
($count_qmark && $count_numbered)
|
||||||
|
) {
|
||||||
|
throw new \OutOfRangeException('Cannot have named, question mark and numbered in the same query', 200);
|
||||||
|
}
|
||||||
|
// // throw if invalid conversion
|
||||||
|
// if (($count_named || $count_qmark) && $convert_to != 'pg') {
|
||||||
|
// throw new \InvalidArgumentException('Cannot convert from named or question mark placeholders to PDO', 300);
|
||||||
|
// }
|
||||||
|
// if ($count_numbered && $convert_to != 'pdo') {
|
||||||
|
// throw new \InvalidArgumentException('Cannot convert from numbered placeholders to Pg', 301);
|
||||||
|
// }
|
||||||
|
// return array
|
||||||
|
$return_placeholders = [
|
||||||
|
// original
|
||||||
|
'original' => [
|
||||||
|
'query' => $query,
|
||||||
|
'params' => $params ?? [],
|
||||||
|
'empty_params' => $params === null ? true : false,
|
||||||
|
],
|
||||||
|
// type found, empty if nothing was done
|
||||||
|
'type' => '',
|
||||||
|
// int: found, not found; -1: problem (set from false)
|
||||||
|
'found' => (int)$found,
|
||||||
|
'matches' => [],
|
||||||
|
// old to new lookup check
|
||||||
|
'params_lookup' => [],
|
||||||
|
// this must match the count in params in new
|
||||||
|
'needed' => 0,
|
||||||
|
// new
|
||||||
|
'query' => '',
|
||||||
|
'params' => [],
|
||||||
|
];
|
||||||
|
// replace basic regex and name settings
|
||||||
|
if ($count_named) {
|
||||||
|
$return_placeholders['type'] = 'named';
|
||||||
|
$return_placeholders['matches'] = $named_matches;
|
||||||
|
$return_placeholders['needed'] = $count_named;
|
||||||
|
} elseif ($count_qmark) {
|
||||||
|
$return_placeholders['type'] = 'question_mark';
|
||||||
|
$return_placeholders['matches'] = $qmark_matches;
|
||||||
|
$return_placeholders['needed'] = $count_qmark;
|
||||||
|
// for each ?:DTN: -> replace with $1 ... $n, any remaining :DTN: remove
|
||||||
|
} elseif ($count_numbered) {
|
||||||
|
$return_placeholders['type'] = 'numbered';
|
||||||
|
$return_placeholders['matches'] = $numbered_matches;
|
||||||
|
$return_placeholders['needed'] = $count_numbered;
|
||||||
|
}
|
||||||
|
// run convert only if matching type and direction
|
||||||
|
if (
|
||||||
|
(($count_named || $count_qmark) && $convert_to == 'pg') ||
|
||||||
|
($count_numbered && $convert_to == 'pdo')
|
||||||
|
) {
|
||||||
|
$param_list = self::updateParamList($return_placeholders);
|
||||||
|
$return_placeholders['params_lookup'] = $param_list['params_lookup'];
|
||||||
|
$return_placeholders['query'] = $param_list['query'];
|
||||||
|
$return_placeholders['params'] = $param_list['params'];
|
||||||
|
}
|
||||||
|
// return data
|
||||||
|
return $return_placeholders;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Updates the params list from one style to the other to match the query output
|
||||||
|
* if original.empty_params is set to true, no params replacement is done
|
||||||
|
* if param replacement has been done in a dbPrepare then this has to be run
|
||||||
|
* with the return palceholders array with params in original filled and empty_params turned off
|
||||||
|
*
|
||||||
|
* phpcs:disable Generic.Files.LineLength
|
||||||
|
* @param array{original:array{query:string,params:array<mixed>,empty_params:bool},type:''|'named'|'numbered'|'question_mark',found:int,matches?:array<string>,params_lookup?:array<mixed>,query?:string,params?:array<mixed>} $converted_placeholders
|
||||||
|
* phpcs:enable Generic.Files.LineLength
|
||||||
|
* @return array{params_lookup:array<mixed>,query:string,params:array<mixed>}
|
||||||
|
*/
|
||||||
|
public static function updateParamList(array $converted_placeholders): array
|
||||||
|
{
|
||||||
|
// skip if nothing set
|
||||||
|
if (!$converted_placeholders['found']) {
|
||||||
|
return [
|
||||||
|
'params_lookup' => [],
|
||||||
|
'query' => '',
|
||||||
|
'params' => []
|
||||||
|
];
|
||||||
|
}
|
||||||
|
$query_new = '';
|
||||||
|
$params_new = [];
|
||||||
|
$params_lookup = [];
|
||||||
|
// set to null if params is empty
|
||||||
|
$params = $converted_placeholders['original']['params'];
|
||||||
|
$empty_params = $converted_placeholders['original']['empty_params'];
|
||||||
|
switch ($converted_placeholders['type']) {
|
||||||
|
case 'named':
|
||||||
|
// 0: full
|
||||||
|
// 0: full
|
||||||
|
// 1: pre part
|
||||||
|
// 2: keep part UNLESS '3' is set
|
||||||
|
// 3: replace part :named
|
||||||
|
$pos = 0;
|
||||||
|
$query_new = preg_replace_callback(
|
||||||
|
self::REGEX_REPLACE_NAMED,
|
||||||
|
function ($matches) use (&$pos, &$params_new, &$params_lookup, $params, $empty_params) {
|
||||||
|
// only count up if $match[3] is not yet in lookup table
|
||||||
|
if (!empty($matches[3]) && empty($params_lookup[$matches[3]])) {
|
||||||
|
$pos++;
|
||||||
|
$params_lookup[$matches[3]] = '$' . $pos;
|
||||||
|
// skip params setup if param list is empty
|
||||||
|
if (!$empty_params) {
|
||||||
|
$params_new[] = $params[$matches[3]] ??
|
||||||
|
throw new \RuntimeException(
|
||||||
|
'Cannot lookup ' . $matches[3] . ' in params list',
|
||||||
|
210
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// add the connectors back (1), and the data sets only if no replacement will be done
|
||||||
|
return $matches[1] . (
|
||||||
|
empty($matches[3]) ?
|
||||||
|
$matches[2] :
|
||||||
|
$params_lookup[$matches[3]] ??
|
||||||
|
throw new \RuntimeException(
|
||||||
|
'Cannot lookup ' . $matches[3] . ' in params lookup list',
|
||||||
|
211
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
$converted_placeholders['original']['query']
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case 'question_mark':
|
||||||
|
if (!$empty_params) {
|
||||||
|
// order and data stays the same
|
||||||
|
$params_new = $params ?? [];
|
||||||
|
}
|
||||||
|
// 0: full
|
||||||
|
// 1: pre part
|
||||||
|
// 2: keep part UNLESS '3' is set
|
||||||
|
// 3: replace part ?
|
||||||
|
$pos = 0;
|
||||||
|
$query_new = preg_replace_callback(
|
||||||
|
self::REGEX_REPLACE_QUESTION_MARK,
|
||||||
|
function ($matches) use (&$pos, &$params_lookup) {
|
||||||
|
// only count pos up for actual replacements we will do
|
||||||
|
if (!empty($matches[3])) {
|
||||||
|
$pos++;
|
||||||
|
$params_lookup[] = '$' . $pos;
|
||||||
|
}
|
||||||
|
// add the connectors back (1), and the data sets only if no replacement will be done
|
||||||
|
return $matches[1] . (
|
||||||
|
empty($matches[3]) ?
|
||||||
|
$matches[2] :
|
||||||
|
'$' . $pos
|
||||||
|
);
|
||||||
|
},
|
||||||
|
$converted_placeholders['original']['query']
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case 'numbered':
|
||||||
|
// 0: full
|
||||||
|
// 1: pre part
|
||||||
|
// 2: keep part UNLESS '3' is set
|
||||||
|
// 3: replace part $numbered
|
||||||
|
$pos = 0;
|
||||||
|
$query_new = preg_replace_callback(
|
||||||
|
self::REGEX_REPLACE_NUMBERED,
|
||||||
|
function ($matches) use (&$pos, &$params_new, &$params_lookup, $params, $empty_params) {
|
||||||
|
// only count up if $match[3] is not yet in lookup table
|
||||||
|
if (!empty($matches[3]) && empty($params_lookup[$matches[3]])) {
|
||||||
|
$pos++;
|
||||||
|
$params_lookup[$matches[3]] = ':' . $pos . '_named';
|
||||||
|
// skip params setup if param list is empty
|
||||||
|
if (!$empty_params) {
|
||||||
|
$params_new[] = $params[($pos - 1)] ??
|
||||||
|
throw new \RuntimeException(
|
||||||
|
'Cannot lookup ' . ($pos - 1) . ' in params list',
|
||||||
|
220
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// add the connectors back (1), and the data sets only if no replacement will be done
|
||||||
|
return $matches[1] . (
|
||||||
|
empty($matches[3]) ?
|
||||||
|
$matches[2] :
|
||||||
|
$params_lookup[$matches[3]] ??
|
||||||
|
throw new \RuntimeException(
|
||||||
|
'Cannot lookup ' . $matches[3] . ' in params lookup list',
|
||||||
|
221
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
$converted_placeholders['original']['query']
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return [
|
||||||
|
'params_lookup' => $params_lookup,
|
||||||
|
'query' => $query_new ?? '',
|
||||||
|
'params' => $params_new,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
@@ -295,8 +295,7 @@ class Support
|
|||||||
* Will start with start_level to skip unwanted from stack
|
* Will start with start_level to skip unwanted from stack
|
||||||
* Defaults to skip level 0 wich is this methid
|
* Defaults to skip level 0 wich is this methid
|
||||||
*
|
*
|
||||||
* @param integer $start_level From what level on, as defaul starts with 1
|
* @param integer $start_level [=1] From what level on, starts with 1 to exclude self
|
||||||
* to exclude self
|
|
||||||
* @return array<mixed> All method names in list where max is last called
|
* @return array<mixed> All method names in list where max is last called
|
||||||
*/
|
*/
|
||||||
public static function getCallerMethodList(int $start_level = 1): array
|
public static function getCallerMethodList(int $start_level = 1): array
|
||||||
@@ -304,15 +303,46 @@ class Support
|
|||||||
$traces = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS);
|
$traces = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS);
|
||||||
$methods = [];
|
$methods = [];
|
||||||
foreach ($traces as $level => $data) {
|
foreach ($traces as $level => $data) {
|
||||||
if ($level >= $start_level) {
|
if ($level < $start_level) {
|
||||||
if (!empty($data['function'])) {
|
continue;
|
||||||
array_unshift($methods, $data['function']);
|
}
|
||||||
}
|
if (!empty($data['function'])) {
|
||||||
|
array_unshift($methods, $data['function']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return $methods;
|
return $methods;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the full call stack from a certain starting level
|
||||||
|
* The return string is
|
||||||
|
* file:line:class->method
|
||||||
|
*
|
||||||
|
* Note that '::' is used for static calls
|
||||||
|
*
|
||||||
|
* @param int $start_level [=1] starts with 1 to exclude itself
|
||||||
|
* @return array<string> string with file, line, class and method
|
||||||
|
*/
|
||||||
|
public static function getCallStack(int $start_level = 1): array
|
||||||
|
{
|
||||||
|
$call_stack = [];
|
||||||
|
$backtrace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS);
|
||||||
|
foreach ($backtrace as $level => $call_trace) {
|
||||||
|
if ($level < $start_level) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$call_stack[] =
|
||||||
|
($call_trace['file'] ?? 'n/f') . ':'
|
||||||
|
. ($call_trace['line'] ?? '-') . ':'
|
||||||
|
. (!empty($call_trace['class']) ?
|
||||||
|
$call_trace['class'] . ($call_trace['type'] ?? '') :
|
||||||
|
''
|
||||||
|
)
|
||||||
|
. $call_trace['function'];
|
||||||
|
}
|
||||||
|
return $call_stack;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the current class where this function is called
|
* Get the current class where this function is called
|
||||||
* Is mostly used in debug log statements to get the class where the debug
|
* Is mostly used in debug log statements to get the class where the debug
|
||||||
|
|||||||
@@ -116,6 +116,29 @@ class System
|
|||||||
3
|
3
|
||||||
) === 'cli' ? true : false;
|
) === 'cli' ? true : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Collect all IP addresses
|
||||||
|
* REMOTE_ADDR, HTTP_X_FORWARD_FOR, CLIENT_IP
|
||||||
|
* and retuns them in an array with index of io source
|
||||||
|
* if address source has addresses with "," will add "-array" with these as array block
|
||||||
|
*
|
||||||
|
* @return array<string,string|array<string>>
|
||||||
|
*/
|
||||||
|
public static function getIpAddresses(): array
|
||||||
|
{
|
||||||
|
$ip_addr = [];
|
||||||
|
foreach (['REMOTE_ADDR', 'HTTP_X_FORWARDED_FOR', 'CLIENT_IP'] as $_ip_source) {
|
||||||
|
if (!empty($_SERVER[$_ip_source])) {
|
||||||
|
$ip_addr[$_ip_source] = $_SERVER[$_ip_source];
|
||||||
|
// same level as ARRAY IF there is a , inside
|
||||||
|
if (strstr($_SERVER[$_ip_source], ',') !== false) {
|
||||||
|
$ip_addr[$_ip_source . '-array'] = explode(',', $_SERVER[$_ip_source]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $ip_addr;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ class CachedFileReader extends \CoreLibs\Language\Core\StringReader
|
|||||||
if (!is_resource($fd)) {
|
if (!is_resource($fd)) {
|
||||||
$this->error = 3; // Cannot read file, probably permissions
|
$this->error = 3; // Cannot read file, probably permissions
|
||||||
} else {
|
} else {
|
||||||
$this->fd_str = fread($fd, filesize($filename) ?: 0) ?: '';
|
$this->fd_str = fread($fd, filesize($filename) ?: 1) ?: '';
|
||||||
fclose($fd);
|
fclose($fd);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -190,7 +190,6 @@ class GetTextReader
|
|||||||
private function loadTables(): void
|
private function loadTables(): void
|
||||||
{
|
{
|
||||||
if (
|
if (
|
||||||
is_array($this->cache_translations) &&
|
|
||||||
is_array($this->table_originals) &&
|
is_array($this->table_originals) &&
|
||||||
is_array($this->table_translations)
|
is_array($this->table_translations)
|
||||||
) {
|
) {
|
||||||
@@ -318,10 +317,7 @@ class GetTextReader
|
|||||||
|
|
||||||
if ($this->enable_cache) {
|
if ($this->enable_cache) {
|
||||||
// Caching enabled, get translated string from cache
|
// Caching enabled, get translated string from cache
|
||||||
if (
|
if (array_key_exists($string, $this->cache_translations)) {
|
||||||
is_array($this->cache_translations) &&
|
|
||||||
array_key_exists($string, $this->cache_translations)
|
|
||||||
) {
|
|
||||||
return $this->cache_translations[$string];
|
return $this->cache_translations[$string];
|
||||||
} else {
|
} else {
|
||||||
return $string;
|
return $string;
|
||||||
@@ -481,7 +477,7 @@ class GetTextReader
|
|||||||
$key = $single . chr(0) . $plural;
|
$key = $single . chr(0) . $plural;
|
||||||
|
|
||||||
if ($this->enable_cache) {
|
if ($this->enable_cache) {
|
||||||
if (is_array($this->cache_translations) && !array_key_exists($key, $this->cache_translations)) {
|
if (!array_key_exists($key, $this->cache_translations)) {
|
||||||
return ($number != 1) ? $plural : $single;
|
return ($number != 1) ? $plural : $single;
|
||||||
} else {
|
} else {
|
||||||
$result = $this->cache_translations[$key];
|
$result = $this->cache_translations[$key];
|
||||||
|
|||||||
@@ -128,7 +128,7 @@ class GetLocale
|
|||||||
$matches
|
$matches
|
||||||
)
|
)
|
||||||
) {
|
) {
|
||||||
$lang = ($matches['lang'] ?? 'en')
|
$lang = $matches['lang']
|
||||||
// add country only if set
|
// add country only if set
|
||||||
. (!empty($matches['country']) ? '_' . $matches['country'] : '');
|
. (!empty($matches['country']) ? '_' . $matches['country'] : '');
|
||||||
} else {
|
} else {
|
||||||
@@ -235,7 +235,7 @@ class GetLocale
|
|||||||
$matches
|
$matches
|
||||||
)
|
)
|
||||||
) {
|
) {
|
||||||
$lang = ($matches['lang'] ?? 'en')
|
$lang = $matches['lang']
|
||||||
// add country only if set
|
// add country only if set
|
||||||
. (!empty($matches['country']) ? '_' . $matches['country'] : '');
|
. (!empty($matches['country']) ? '_' . $matches['country'] : '');
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -24,16 +24,20 @@ class ErrorMessage
|
|||||||
|
|
||||||
/** @var bool $log_error global flag to log error level message */
|
/** @var bool $log_error global flag to log error level message */
|
||||||
private bool $log_error = false;
|
private bool $log_error = false;
|
||||||
|
/** @var bool $log_warning global flat to log warning level messages */
|
||||||
|
private bool $log_warning = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* init ErrorMessage
|
* init ErrorMessage
|
||||||
*
|
*
|
||||||
* @param \CoreLibs\Logging\Logging $log
|
* @param \CoreLibs\Logging\Logging $log
|
||||||
* @param null|bool $log_error [=null], defaults to false if log is not level debug
|
* @param null|bool $log_error [=null], defaults to false if log is not level debug
|
||||||
|
* @param null|bool $log_warning [=null], defaults to false if log is not level debug
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(
|
||||||
\CoreLibs\Logging\Logging $log,
|
\CoreLibs\Logging\Logging $log,
|
||||||
?bool $log_error = null
|
?bool $log_error = null,
|
||||||
|
?bool $log_warning = null
|
||||||
) {
|
) {
|
||||||
$this->log = $log;
|
$this->log = $log;
|
||||||
// if log default logging is debug then log_error is default set to true
|
// if log default logging is debug then log_error is default set to true
|
||||||
@@ -43,6 +47,13 @@ class ErrorMessage
|
|||||||
$log_error = $log_error ?? false;
|
$log_error = $log_error ?? false;
|
||||||
}
|
}
|
||||||
$this->log_error = $log_error;
|
$this->log_error = $log_error;
|
||||||
|
// if log default logging is debug then log_warning is default set to true
|
||||||
|
if ($this->log->loggingLevelIsDebug() && $log_warning === null) {
|
||||||
|
$log_warning = true;
|
||||||
|
} else {
|
||||||
|
$log_warning = $log_warning ?? false;
|
||||||
|
}
|
||||||
|
$this->log_warning = $log_warning;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -81,6 +92,8 @@ class ErrorMessage
|
|||||||
* @param array<mixed> $context Additionl info for abort/crash messages
|
* @param array<mixed> $context Additionl info for abort/crash messages
|
||||||
* @param bool|null $log_error [=null] log level 'error' to error, if null use global,
|
* @param bool|null $log_error [=null] log level 'error' to error, if null use global,
|
||||||
* else set for this call only
|
* else set for this call only
|
||||||
|
* @param bool|null $log_warning [=null] log level 'warning' to warning, if null use global,
|
||||||
|
* else set for this call only
|
||||||
*/
|
*/
|
||||||
public function setErrorMsg(
|
public function setErrorMsg(
|
||||||
string $error_id,
|
string $error_id,
|
||||||
@@ -93,10 +106,14 @@ class ErrorMessage
|
|||||||
?string $message = null,
|
?string $message = null,
|
||||||
array $context = [],
|
array $context = [],
|
||||||
?bool $log_error = null,
|
?bool $log_error = null,
|
||||||
|
?bool $log_warning = null,
|
||||||
): void {
|
): void {
|
||||||
if ($log_error === null) {
|
if ($log_error === null) {
|
||||||
$log_error = $this->log_error;
|
$log_error = $this->log_error;
|
||||||
}
|
}
|
||||||
|
if ($log_warning === null) {
|
||||||
|
$log_warning = $this->log_warning;
|
||||||
|
}
|
||||||
$original_level = $level;
|
$original_level = $level;
|
||||||
$level = MessageLevel::fromName($level)->name;
|
$level = MessageLevel::fromName($level)->name;
|
||||||
// if not string set, write message string if set, else level/error id
|
// if not string set, write message string if set, else level/error id
|
||||||
@@ -121,6 +138,14 @@ class ErrorMessage
|
|||||||
'level' => $original_level,
|
'level' => $original_level,
|
||||||
], $context));
|
], $context));
|
||||||
break;
|
break;
|
||||||
|
case 'warn':
|
||||||
|
if ($log_warning) {
|
||||||
|
$this->log->warning($message ?? $str, array_merge([
|
||||||
|
'id' => $error_id,
|
||||||
|
'level' => $original_level,
|
||||||
|
], $context));
|
||||||
|
}
|
||||||
|
break;
|
||||||
case 'error':
|
case 'error':
|
||||||
if ($log_error) {
|
if ($log_error) {
|
||||||
$this->log->error($message ?? $str, array_merge([
|
$this->log->error($message ?? $str, array_merge([
|
||||||
@@ -169,6 +194,8 @@ class ErrorMessage
|
|||||||
* @param array<mixed> $context Additionl info for abort/crash messages
|
* @param array<mixed> $context Additionl info for abort/crash messages
|
||||||
* @param bool|null $log_error [=null] log level 'error' to error, if null use global,
|
* @param bool|null $log_error [=null] log level 'error' to error, if null use global,
|
||||||
* else set for this call only
|
* else set for this call only
|
||||||
|
* @param bool|null $log_warning [=null] log level 'warning' to warning, if null use global,
|
||||||
|
* else set for this call only
|
||||||
*/
|
*/
|
||||||
public function setMessage(
|
public function setMessage(
|
||||||
string $level,
|
string $level,
|
||||||
@@ -181,6 +208,7 @@ class ErrorMessage
|
|||||||
?string $message = null,
|
?string $message = null,
|
||||||
array $context = [],
|
array $context = [],
|
||||||
?bool $log_error = null,
|
?bool $log_error = null,
|
||||||
|
?bool $log_warning = null,
|
||||||
): void {
|
): void {
|
||||||
$this->setErrorMsg(
|
$this->setErrorMsg(
|
||||||
$error_id ?? '',
|
$error_id ?? '',
|
||||||
@@ -192,7 +220,8 @@ class ErrorMessage
|
|||||||
$jump_target,
|
$jump_target,
|
||||||
$message,
|
$message,
|
||||||
$context,
|
$context,
|
||||||
$log_error
|
$log_error,
|
||||||
|
$log_warning
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -314,6 +343,27 @@ class ErrorMessage
|
|||||||
{
|
{
|
||||||
return $this->log_error;
|
return $this->log_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the log warning flag
|
||||||
|
*
|
||||||
|
* @param bool $flag True to log level warning too, False for do not (Default)
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function setFlagLogWarning(bool $flag): void
|
||||||
|
{
|
||||||
|
$this->log_warning = $flag;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the current log error flag
|
||||||
|
*
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function getFlagLogWarning(): bool
|
||||||
|
{
|
||||||
|
return $this->log_warning;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ namespace CoreLibs\Logging\Logger;
|
|||||||
enum MessageLevel: int
|
enum MessageLevel: int
|
||||||
{
|
{
|
||||||
case ok = 100;
|
case ok = 100;
|
||||||
|
case success = 150; // special for file uploads
|
||||||
case info = 200;
|
case info = 200;
|
||||||
case notice = 250;
|
case notice = 250;
|
||||||
case warn = 300;
|
case warn = 300;
|
||||||
@@ -30,6 +31,7 @@ enum MessageLevel: int
|
|||||||
{
|
{
|
||||||
return match (strtolower($name)) {
|
return match (strtolower($name)) {
|
||||||
'ok' => self::ok,
|
'ok' => self::ok,
|
||||||
|
'success' => self::success,
|
||||||
'info' => self::info,
|
'info' => self::info,
|
||||||
'notice' => self::notice,
|
'notice' => self::notice,
|
||||||
'warn', 'warning' => self::warn,
|
'warn', 'warning' => self::warn,
|
||||||
|
|||||||
@@ -194,13 +194,13 @@ class Elements
|
|||||||
"/(mailto:)?(\>)?\b([\w\.-]+)@([\w\.\-]+)\.([a-zA-Z]{2,4})\b(\|([^\||^#]+)(#([^\|]+))?\|)?/",
|
"/(mailto:)?(\>)?\b([\w\.-]+)@([\w\.\-]+)\.([a-zA-Z]{2,4})\b(\|([^\||^#]+)(#([^\|]+))?\|)?/",
|
||||||
function ($matches) {
|
function ($matches) {
|
||||||
return self::createEmail(
|
return self::createEmail(
|
||||||
$matches[1] ?? '',
|
$matches[1],
|
||||||
$matches[2] ?? '',
|
$matches[2],
|
||||||
$matches[3] ?? '',
|
$matches[3],
|
||||||
$matches[4] ?? '',
|
$matches[4],
|
||||||
$matches[5] ?? '',
|
$matches[5],
|
||||||
$matches[7] ?? '',
|
$matches[7] ?? '',
|
||||||
$matches[9] ?? ''
|
$matches[9] ?? '',
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
$output
|
$output
|
||||||
|
|||||||
@@ -310,7 +310,7 @@ class Generate
|
|||||||
* construct form generator
|
* construct form generator
|
||||||
*
|
*
|
||||||
* phpcs:ignore
|
* phpcs:ignore
|
||||||
* @param array{db_name:string,db_user:string,db_pass:string,db_host:string,db_port:int,db_schema:string,db_encoding:string,db_type:string,db_ssl:string,db_convert_type?:string[]} $db_config db config array, mandatory
|
* @param array{db_name:string,db_user:string,db_pass:string,db_host:string,db_port:int,db_schema:string,db_encoding:string,db_type:string,db_ssl:string,db_convert_type?:string[],db_convert_placeholder?:bool,db_convert_placeholder_target?:string,db_debug_replace_placeholder?:bool} $db_config db config array, mandatory
|
||||||
* @param \CoreLibs\Logging\Logging $log Logging class
|
* @param \CoreLibs\Logging\Logging $log Logging class
|
||||||
* @param \CoreLibs\Language\L10n $l10n l10n language class
|
* @param \CoreLibs\Language\L10n $l10n l10n language class
|
||||||
* @param array<string,mixed> $login_acl Login ACL array,
|
* @param array<string,mixed> $login_acl Login ACL array,
|
||||||
@@ -474,7 +474,7 @@ class Generate
|
|||||||
$page_name_camel_case
|
$page_name_camel_case
|
||||||
);
|
);
|
||||||
try {
|
try {
|
||||||
/** @var TableArrays\Interface\TableArraysInterface|false $class */
|
/** @var TableArrays\Interface\TableArraysInterface $class */
|
||||||
$class = new $class_string($this);
|
$class = new $class_string($this);
|
||||||
} catch (\Throwable $t) {
|
} catch (\Throwable $t) {
|
||||||
$this->log->critical('CLASS LOADING: Failed loading: ' . $class_string . ' => ' . $t->getMessage());
|
$this->log->critical('CLASS LOADING: Failed loading: ' . $class_string . ' => ' . $t->getMessage());
|
||||||
@@ -826,27 +826,28 @@ class Generate
|
|||||||
$pk_selected = $res[$this->int_pk_name];
|
$pk_selected = $res[$this->int_pk_name];
|
||||||
}
|
}
|
||||||
$t_string = '';
|
$t_string = '';
|
||||||
foreach ($this->field_array as $i => $field_array) {
|
foreach ($this->field_array as $field_array) {
|
||||||
if ($t_string) {
|
if ($t_string) {
|
||||||
$t_string .= ', ';
|
$t_string .= ', ';
|
||||||
}
|
}
|
||||||
if (isset($field_array['before_value'])) {
|
if (!empty($field_array['before_value'])) {
|
||||||
$t_string .= $field_array['before_value'];
|
$t_string .= $this->l->__($field_array['before_value']);
|
||||||
}
|
}
|
||||||
// must have res element set
|
// must have res element set
|
||||||
if (
|
if (
|
||||||
isset($field_array['name']) &&
|
!empty($field_array['name']) &&
|
||||||
isset($res[$field_array['name']])
|
isset($res[$field_array['name']])
|
||||||
) {
|
) {
|
||||||
if (isset($field_array['binary'])) {
|
$_t_value = '';
|
||||||
if (isset($field_array['binary'][0])) {
|
// if we have a binary set, where 0 = YES and 1 = NO
|
||||||
$t_string .= $field_array['binary'][0];
|
if (!empty($field_array['binary'])) {
|
||||||
} elseif (isset($field_array['binary'][1])) {
|
$_t_value = !empty($res[$field_array['name']]) ?
|
||||||
$t_string .= $field_array['binary'][1];
|
($field_array['binary'][0] ?? 'Yes') :
|
||||||
}
|
($field_array['binary'][1] ?? 'No');
|
||||||
} else {
|
} else {
|
||||||
$t_string .= $res[$field_array['name']];
|
$_t_value = $res[$field_array['name']];
|
||||||
}
|
}
|
||||||
|
$t_string .= $this->l->__($_t_value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$pk_names[] = $t_string;
|
$pk_names[] = $t_string;
|
||||||
@@ -1506,7 +1507,7 @@ class Generate
|
|||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
!empty($this->reference_array[$key]['mandatory']) &&
|
!empty($this->reference_array[$key]['mandatory']) &&
|
||||||
!$this->reference_array[$key]['selected'][0]
|
empty($this->reference_array[$key]['selected'][0])
|
||||||
) {
|
) {
|
||||||
$this->msg .= sprintf(
|
$this->msg .= sprintf(
|
||||||
$this->l->__('Please select at least one Element from field <b>%s</b>!<br>'),
|
$this->l->__('Please select at least one Element from field <b>%s</b>!<br>'),
|
||||||
@@ -1756,14 +1757,9 @@ class Generate
|
|||||||
$this->dba->setTableArrayEntry($this->dba->getTableArray()[$key]['preset'], $key, 'value');
|
$this->dba->setTableArrayEntry($this->dba->getTableArray()[$key]['preset'], $key, 'value');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (is_array($this->reference_array)) {
|
reset($this->reference_array);
|
||||||
if (!is_array($this->reference_array)) {
|
foreach ($this->reference_array as $key => $value) {
|
||||||
$this->reference_array = [];
|
unset($this->reference_array[$key]['selected']);
|
||||||
}
|
|
||||||
reset($this->reference_array);
|
|
||||||
foreach ($this->reference_array as $key => $value) {
|
|
||||||
unset($this->reference_array[$key]['selected']);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
$this->warning = 1;
|
$this->warning = 1;
|
||||||
$this->msg = $this->l->__('Cleared for new Dataset!');
|
$this->msg = $this->l->__('Cleared for new Dataset!');
|
||||||
@@ -1786,20 +1782,15 @@ class Generate
|
|||||||
$this->dba->unsetTableArrayEntry($key, 'input_value');
|
$this->dba->unsetTableArrayEntry($key, 'input_value');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_array($this->reference_array)) {
|
// load each reference_table
|
||||||
// load each reference_table
|
reset($this->reference_array);
|
||||||
if (!is_array($this->reference_array)) {
|
foreach ($this->reference_array as $key => $value) {
|
||||||
$this->reference_array = [];
|
unset($this->reference_array[$key]['selected']);
|
||||||
}
|
$q = 'SELECT ' . $this->reference_array[$key]['other_table_pk']
|
||||||
reset($this->reference_array);
|
. ' FROM ' . $this->reference_array[$key]['table_name']
|
||||||
foreach ($this->reference_array as $key => $value) {
|
. ' WHERE ' . $this->int_pk_name . ' = ' . $this->dba->getTableArray()[$this->int_pk_name]['value'];
|
||||||
unset($this->reference_array[$key]['selected']);
|
while (is_array($res = $this->dba->dbReturn($q))) {
|
||||||
$q = 'SELECT ' . $this->reference_array[$key]['other_table_pk']
|
$this->reference_array[$key]['selected'][] = $res[$this->reference_array[$key]['other_table_pk']];
|
||||||
. ' FROM ' . $this->reference_array[$key]['table_name']
|
|
||||||
. ' WHERE ' . $this->int_pk_name . ' = ' . $this->dba->getTableArray()[$this->int_pk_name]['value'];
|
|
||||||
while (is_array($res = $this->dba->dbReturn($q))) {
|
|
||||||
$this->reference_array[$key]['selected'][] = $res[$this->reference_array[$key]['other_table_pk']];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$this->warning = 1;
|
$this->warning = 1;
|
||||||
@@ -1978,24 +1969,19 @@ class Generate
|
|||||||
// write the object
|
// write the object
|
||||||
$this->dba->dbWrite($addslashes, [], true);
|
$this->dba->dbWrite($addslashes, [], true);
|
||||||
// write reference array (s) if necessary
|
// write reference array (s) if necessary
|
||||||
if (is_array($this->reference_array)) {
|
reset($this->reference_array);
|
||||||
if (!is_array($this->reference_array)) {
|
foreach ($this->reference_array as $reference_array) {
|
||||||
$this->reference_array = [];
|
$q = 'DELETE FROM ' . $reference_array['table_name']
|
||||||
|
. ' WHERE ' . $this->int_pk_name . ' = ' . $this->dba->getTableArray()[$this->int_pk_name]['value'];
|
||||||
|
$this->dba->dbExec($q);
|
||||||
|
$q = 'INSERT INTO ' . $reference_array['table_name']
|
||||||
|
. ' (' . $reference_array['other_table_pk'] . ', ' . $this->int_pk_name . ') VALUES ';
|
||||||
|
for ($i = 0, $i_max = count($reference_array['selected']); $i < $i_max; $i++) {
|
||||||
|
$t_q = '(' . $reference_array['selected'][$i] . ', '
|
||||||
|
. $this->dba->getTableArray()[$this->int_pk_name]['value'] . ')';
|
||||||
|
$this->dba->dbExec($q . $t_q);
|
||||||
}
|
}
|
||||||
reset($this->reference_array);
|
} // foreach reference arrays
|
||||||
foreach ($this->reference_array as $reference_array) {
|
|
||||||
$q = 'DELETE FROM ' . $reference_array['table_name']
|
|
||||||
. ' WHERE ' . $this->int_pk_name . ' = ' . $this->dba->getTableArray()[$this->int_pk_name]['value'];
|
|
||||||
$this->dba->dbExec($q);
|
|
||||||
$q = 'INSERT INTO ' . $reference_array['table_name']
|
|
||||||
. ' (' . $reference_array['other_table_pk'] . ', ' . $this->int_pk_name . ') VALUES ';
|
|
||||||
for ($i = 0, $i_max = count($reference_array['selected']); $i < $i_max; $i++) {
|
|
||||||
$t_q = '(' . $reference_array['selected'][$i] . ', '
|
|
||||||
. $this->dba->getTableArray()[$this->int_pk_name]['value'] . ')';
|
|
||||||
$this->dba->dbExec($q . $t_q);
|
|
||||||
}
|
|
||||||
} // foreach reference arrays
|
|
||||||
} // if reference arrays
|
|
||||||
// write element list
|
// write element list
|
||||||
if (!empty($this->element_list)) {
|
if (!empty($this->element_list)) {
|
||||||
$type = [];
|
$type = [];
|
||||||
@@ -2229,16 +2215,11 @@ class Generate
|
|||||||
public function formDeleteTableArray()
|
public function formDeleteTableArray()
|
||||||
{
|
{
|
||||||
// remove any reference arrays
|
// remove any reference arrays
|
||||||
if (is_array($this->reference_array)) {
|
reset($this->reference_array);
|
||||||
if (!is_array($this->reference_array)) {
|
foreach ($this->reference_array as $reference_array) {
|
||||||
$this->reference_array = [];
|
$q = 'DELETE FROM ' . $reference_array['table_name']
|
||||||
}
|
. ' WHERE ' . $this->int_pk_name . ' = ' . $this->dba->getTableArray()[$this->int_pk_name]['value'];
|
||||||
reset($this->reference_array);
|
$this->dba->dbExec($q);
|
||||||
foreach ($this->reference_array as $reference_array) {
|
|
||||||
$q = 'DELETE FROM ' . $reference_array['table_name']
|
|
||||||
. ' WHERE ' . $this->int_pk_name . ' = ' . $this->dba->getTableArray()[$this->int_pk_name]['value'];
|
|
||||||
$this->dba->dbExec($q);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// remove any element list references
|
// remove any element list references
|
||||||
if (!empty($this->element_list)) {
|
if (!empty($this->element_list)) {
|
||||||
|
|||||||
@@ -53,6 +53,7 @@ class EditPages implements Interface\TableArraysInterface
|
|||||||
'value' => $_POST['name'] ?? '',
|
'value' => $_POST['name'] ?? '',
|
||||||
'output_name' => 'Page name',
|
'output_name' => 'Page name',
|
||||||
'mandatory' => 1,
|
'mandatory' => 1,
|
||||||
|
'error_check' => 'unique',
|
||||||
'type' => 'text'
|
'type' => 'text'
|
||||||
],
|
],
|
||||||
'order_number' => [
|
'order_number' => [
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* sets a form token in the _SESSION variable
|
* sets a form token in the _SESSION variable
|
||||||
* session must be started for this to work
|
* session must be started and running for this to work
|
||||||
*/
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|||||||
@@ -78,7 +78,7 @@ class Image
|
|||||||
if (!empty($dummy) && file_exists($filename) && is_file($filename)) {
|
if (!empty($dummy) && file_exists($filename) && is_file($filename)) {
|
||||||
$return_data = $filename;
|
$return_data = $filename;
|
||||||
} else {
|
} else {
|
||||||
throw new \Exception('Could not set dummy return file: ' . $dummy . ' in ' . $filename);
|
throw new \RuntimeException('Could not set dummy return file: ' . $dummy . ' in ' . $filename);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
$return_data = $dummy;
|
$return_data = $dummy;
|
||||||
@@ -204,11 +204,11 @@ class Image
|
|||||||
E_USER_DEPRECATED
|
E_USER_DEPRECATED
|
||||||
);
|
);
|
||||||
// NOTE: we need to depracte this
|
// NOTE: we need to depracte this
|
||||||
$cache_folder = BASE . LAYOUT . CONTENT_PATH . CACHE . IMAGES;
|
$cache_folder = BASE . CONTENT_PATH . LAYOUT . CACHE . IMAGES;
|
||||||
$web_folder = LAYOUT . CACHE . IMAGES;
|
$web_folder = LAYOUT . CACHE . IMAGES;
|
||||||
if (!is_dir($cache_folder)) {
|
if (!is_dir($cache_folder)) {
|
||||||
if (false === mkdir($cache_folder)) {
|
if (false === mkdir($cache_folder)) {
|
||||||
$cache_folder = BASE . LAYOUT . CONTENT_PATH . CACHE;
|
$cache_folder = BASE . CONTENT_PATH . LAYOUT . CACHE;
|
||||||
$web_folder = LAYOUT . CACHE;
|
$web_folder = LAYOUT . CACHE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -256,8 +256,8 @@ class Image
|
|||||||
}
|
}
|
||||||
// check resize parameters
|
// check resize parameters
|
||||||
if ($inc_width > $thumb_width || $inc_height > $thumb_height) {
|
if ($inc_width > $thumb_width || $inc_height > $thumb_height) {
|
||||||
$thumb_width_r = 0;
|
$thumb_width_r = 1;
|
||||||
$thumb_height_r = 0;
|
$thumb_height_r = 1;
|
||||||
// we need to keep the aspect ration on longest side
|
// we need to keep the aspect ration on longest side
|
||||||
if (
|
if (
|
||||||
($inc_height > $inc_width &&
|
($inc_height > $inc_width &&
|
||||||
@@ -288,6 +288,12 @@ class Image
|
|||||||
!file_exists($thumbnail_write_path . $thumbnail)
|
!file_exists($thumbnail_write_path . $thumbnail)
|
||||||
) {
|
) {
|
||||||
// image, copy source image, offset in image, source x/y, new size, source image size
|
// image, copy source image, offset in image, source x/y, new size, source image size
|
||||||
|
if ($thumb_width_r < 1) {
|
||||||
|
$thumb_width_r = 1;
|
||||||
|
}
|
||||||
|
if ($thumb_height_r < 1) {
|
||||||
|
$thumb_height_r = 1;
|
||||||
|
}
|
||||||
$thumb = imagecreatetruecolor($thumb_width_r, $thumb_height_r);
|
$thumb = imagecreatetruecolor($thumb_width_r, $thumb_height_r);
|
||||||
if ($thumb === false) {
|
if ($thumb === false) {
|
||||||
throw new \RuntimeException(
|
throw new \RuntimeException(
|
||||||
@@ -380,9 +386,7 @@ class Image
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// add output path
|
// add output path
|
||||||
if ($thumbnail !== false) {
|
$thumbnail = $thumbnail_web_path . $thumbnail;
|
||||||
$thumbnail = $thumbnail_web_path . $thumbnail;
|
|
||||||
}
|
|
||||||
} elseif ($create_dummy === true) {
|
} elseif ($create_dummy === true) {
|
||||||
// create dummy image in the thumbnail size
|
// create dummy image in the thumbnail size
|
||||||
// if one side is missing, use the other side to create a square
|
// if one side is missing, use the other side to create a square
|
||||||
@@ -399,10 +403,10 @@ class Image
|
|||||||
!file_exists($thumbnail_write_path . $thumbnail)
|
!file_exists($thumbnail_write_path . $thumbnail)
|
||||||
) {
|
) {
|
||||||
// if both are unset, set to 250
|
// if both are unset, set to 250
|
||||||
if ($thumb_height == 0) {
|
if ($thumb_height < 1) {
|
||||||
$thumb_height = 250;
|
$thumb_height = 250;
|
||||||
}
|
}
|
||||||
if ($thumb_width == 0) {
|
if ($thumb_width < 1) {
|
||||||
$thumb_width = 250;
|
$thumb_width = 250;
|
||||||
}
|
}
|
||||||
$thumb = imagecreatetruecolor($thumb_width, $thumb_height);
|
$thumb = imagecreatetruecolor($thumb_width, $thumb_height);
|
||||||
|
|||||||
@@ -156,7 +156,7 @@ class ProgressBar
|
|||||||
{
|
{
|
||||||
// avoid divison through 0
|
// avoid divison through 0
|
||||||
if ($this->max - $this->min == 0) {
|
if ($this->max - $this->min == 0) {
|
||||||
$this->max ++;
|
$this->max++;
|
||||||
}
|
}
|
||||||
$percent = round(($step - $this->min) / ($this->max - $this->min) * 100);
|
$percent = round(($step - $this->min) / ($this->max - $this->min) * 100);
|
||||||
if ($percent > 100) {
|
if ($percent > 100) {
|
||||||
@@ -186,7 +186,7 @@ class ProgressBar
|
|||||||
}
|
}
|
||||||
// avoid divison through 0
|
// avoid divison through 0
|
||||||
if ($this->max - $this->min == 0) {
|
if ($this->max - $this->min == 0) {
|
||||||
$this->max ++;
|
$this->max++;
|
||||||
}
|
}
|
||||||
$pixel = round(($step - $this->min) * ($bar - ($this->pedding * 2)) / ($this->max - $this->min));
|
$pixel = round(($step - $this->min) * ($bar - ($this->pedding * 2)) / ($this->max - $this->min));
|
||||||
if ($step <= $this->min) {
|
if ($step <= $this->min) {
|
||||||
|
|||||||
@@ -21,58 +21,82 @@ use SodiumException;
|
|||||||
|
|
||||||
class SymmetricEncryption
|
class SymmetricEncryption
|
||||||
{
|
{
|
||||||
|
/** @var SymmetricEncryption self instance */
|
||||||
|
private static SymmetricEncryption $instance;
|
||||||
|
|
||||||
|
/** @var string bin hex key */
|
||||||
|
private string $key = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Encrypt a message
|
* init class
|
||||||
|
* if key not passed, key must be set with createKey
|
||||||
*
|
*
|
||||||
* @param string $message Message to encrypt
|
* @param string|null|null $key
|
||||||
* @param string $key Encryption key (as hex string)
|
|
||||||
* @return string
|
|
||||||
* @throws \Exception
|
|
||||||
* @throws \RangeException
|
|
||||||
*/
|
*/
|
||||||
public static function encrypt(string $message, string $key): string
|
public function __construct(
|
||||||
|
string|null $key = null
|
||||||
|
) {
|
||||||
|
if ($key != null) {
|
||||||
|
$this->setKey($key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the singleton self object.
|
||||||
|
* For function wrapper use
|
||||||
|
*
|
||||||
|
* @return SymmetricEncryption object
|
||||||
|
*/
|
||||||
|
public static function getInstance(string|null $key = null): self
|
||||||
|
{
|
||||||
|
if (empty(self::$instance)) {
|
||||||
|
self::$instance = new self($key);
|
||||||
|
}
|
||||||
|
return self::$instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ************************************************************************
|
||||||
|
* MARK: PRIVATE
|
||||||
|
* *************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* create key and check validity
|
||||||
|
*
|
||||||
|
* @param string $key The key from which the binary key will be created
|
||||||
|
* @return string Binary key string
|
||||||
|
*/
|
||||||
|
private function createKey(string $key): string
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$key = CreateKey::hex2bin($key);
|
$key = CreateKey::hex2bin($key);
|
||||||
} catch (SodiumException $e) {
|
} catch (SodiumException $e) {
|
||||||
throw new \UnexpectedValueException('Invalid hex key');
|
throw new \UnexpectedValueException('Invalid hex key: ' . $e->getMessage());
|
||||||
}
|
}
|
||||||
if (mb_strlen($key, '8bit') !== SODIUM_CRYPTO_SECRETBOX_KEYBYTES) {
|
if (mb_strlen($key, '8bit') !== SODIUM_CRYPTO_SECRETBOX_KEYBYTES) {
|
||||||
throw new \RangeException(
|
throw new \RangeException(
|
||||||
'Key is not the correct size (must be '
|
'Key is not the correct size (must be '
|
||||||
. 'SODIUM_CRYPTO_SECRETBOX_KEYBYTES bytes long).'
|
. SODIUM_CRYPTO_SECRETBOX_KEYBYTES . ' bytes long).'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
$nonce = random_bytes(SODIUM_CRYPTO_SECRETBOX_NONCEBYTES);
|
return $key;
|
||||||
|
|
||||||
$cipher = base64_encode(
|
|
||||||
$nonce
|
|
||||||
. sodium_crypto_secretbox(
|
|
||||||
$message,
|
|
||||||
$nonce,
|
|
||||||
$key
|
|
||||||
)
|
|
||||||
);
|
|
||||||
sodium_memzero($message);
|
|
||||||
sodium_memzero($key);
|
|
||||||
return $cipher;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decrypt a message
|
* Decryption call
|
||||||
*
|
*
|
||||||
* @param string $encrypted Message encrypted with safeEncrypt()
|
* @param string $encrypted Text to decrypt
|
||||||
* @param string $key Encryption key (as hex string)
|
* @param ?string $key Mandatory encryption key, will throw exception if empty
|
||||||
* @return string
|
* @return string Plain text
|
||||||
* @throws \Exception
|
* @throws \RangeException
|
||||||
|
* @throws \UnexpectedValueException
|
||||||
|
* @throws \UnexpectedValueException
|
||||||
*/
|
*/
|
||||||
public static function decrypt(string $encrypted, string $key): string
|
private function decryptData(string $encrypted, ?string $key): string
|
||||||
{
|
{
|
||||||
try {
|
if (empty($key)) {
|
||||||
$key = CreateKey::hex2bin($key);
|
throw new \UnexpectedValueException('Key not set');
|
||||||
} catch (SodiumException $e) {
|
|
||||||
throw new \Exception('Invalid hex key');
|
|
||||||
}
|
}
|
||||||
|
$key = $this->createKey($key);
|
||||||
$decoded = base64_decode($encrypted);
|
$decoded = base64_decode($encrypted);
|
||||||
$nonce = mb_substr($decoded, 0, SODIUM_CRYPTO_SECRETBOX_NONCEBYTES, '8bit');
|
$nonce = mb_substr($decoded, 0, SODIUM_CRYPTO_SECRETBOX_NONCEBYTES, '8bit');
|
||||||
$ciphertext = mb_substr($decoded, SODIUM_CRYPTO_SECRETBOX_NONCEBYTES, null, '8bit');
|
$ciphertext = mb_substr($decoded, SODIUM_CRYPTO_SECRETBOX_NONCEBYTES, null, '8bit');
|
||||||
@@ -85,7 +109,7 @@ class SymmetricEncryption
|
|||||||
$key
|
$key
|
||||||
);
|
);
|
||||||
} catch (SodiumException $e) {
|
} catch (SodiumException $e) {
|
||||||
throw new \UnexpectedValueException('Invalid ciphertext (too short)');
|
throw new \UnexpectedValueException('Decipher message failed: ' . $e->getMessage());
|
||||||
}
|
}
|
||||||
if (!is_string($plain)) {
|
if (!is_string($plain)) {
|
||||||
throw new \UnexpectedValueException('Invalid Key');
|
throw new \UnexpectedValueException('Invalid Key');
|
||||||
@@ -94,6 +118,117 @@ class SymmetricEncryption
|
|||||||
sodium_memzero($key);
|
sodium_memzero($key);
|
||||||
return $plain;
|
return $plain;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Encrypt a message
|
||||||
|
*
|
||||||
|
* @param string $message Message to encrypt
|
||||||
|
* @param ?string $key Mandatory encryption key, will throw exception if empty
|
||||||
|
* @return string
|
||||||
|
* @throws \Exception
|
||||||
|
* @throws \RangeException
|
||||||
|
*/
|
||||||
|
private function encryptData(string $message, ?string $key): string
|
||||||
|
{
|
||||||
|
if (empty($this->key) || $key === null) {
|
||||||
|
throw new \UnexpectedValueException('Key not set');
|
||||||
|
}
|
||||||
|
$key = $this->createKey($key);
|
||||||
|
$nonce = random_bytes(SODIUM_CRYPTO_SECRETBOX_NONCEBYTES);
|
||||||
|
try {
|
||||||
|
$cipher = base64_encode(
|
||||||
|
$nonce
|
||||||
|
. sodium_crypto_secretbox(
|
||||||
|
$message,
|
||||||
|
$nonce,
|
||||||
|
$key,
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} catch (SodiumException $e) {
|
||||||
|
throw new \UnexpectedValueException("Create encrypted message failed: " . $e->getMessage());
|
||||||
|
}
|
||||||
|
sodium_memzero($message);
|
||||||
|
sodium_memzero($key);
|
||||||
|
return $cipher;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ************************************************************************
|
||||||
|
* MARK: PUBLIC
|
||||||
|
* *************************************************************************/
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set a new key for encryption
|
||||||
|
*
|
||||||
|
* @param string $key
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function setKey(string $key)
|
||||||
|
{
|
||||||
|
if (empty($key)) {
|
||||||
|
throw new \UnexpectedValueException('Key cannot be empty');
|
||||||
|
}
|
||||||
|
$this->key = $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Decrypt a message
|
||||||
|
* static version
|
||||||
|
*
|
||||||
|
* @param string $encrypted Message encrypted with safeEncrypt()
|
||||||
|
* @param string $key Encryption key (as hex string)
|
||||||
|
* @return string
|
||||||
|
* @throws \Exception
|
||||||
|
* @throws \RangeException
|
||||||
|
* @throws \UnexpectedValueException
|
||||||
|
* @throws \UnexpectedValueException
|
||||||
|
*/
|
||||||
|
public static function decryptKey(string $encrypted, string $key): string
|
||||||
|
{
|
||||||
|
return self::getInstance()->decryptData($encrypted, $key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Decrypt a message
|
||||||
|
*
|
||||||
|
* @param string $encrypted Message encrypted with safeEncrypt()
|
||||||
|
* @return string
|
||||||
|
* @throws \RangeException
|
||||||
|
* @throws \UnexpectedValueException
|
||||||
|
* @throws \UnexpectedValueException
|
||||||
|
*/
|
||||||
|
public function decrypt(string $encrypted): string
|
||||||
|
{
|
||||||
|
return $this->decryptData($encrypted, $this->key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Encrypt a message
|
||||||
|
* static version
|
||||||
|
*
|
||||||
|
* @param string $message Message to encrypt
|
||||||
|
* @param string $key Encryption key (as hex string)
|
||||||
|
* @return string
|
||||||
|
* @throws \Exception
|
||||||
|
* @throws \RangeException
|
||||||
|
*/
|
||||||
|
public static function encryptKey(string $message, string $key): string
|
||||||
|
{
|
||||||
|
return self::getInstance()->encryptData($message, $key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Encrypt a message
|
||||||
|
*
|
||||||
|
* @param string $message Message to encrypt
|
||||||
|
* @return string
|
||||||
|
* @throws \Exception
|
||||||
|
* @throws \RangeException
|
||||||
|
*/
|
||||||
|
public function encrypt(string $message): string
|
||||||
|
{
|
||||||
|
return $this->encryptData($message, $this->key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ use CoreLibs\Template\HtmlBuilder\General\HtmlBuilderExcpetion;
|
|||||||
class Block
|
class Block
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Create Element
|
||||||
*
|
*
|
||||||
* @param string $tag
|
* @param string $tag
|
||||||
* @param string $id
|
* @param string $id
|
||||||
@@ -86,7 +86,7 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Add multiple elements to the base element
|
||||||
*
|
*
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $base
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $base
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} ...$attach
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} ...$attach
|
||||||
@@ -101,7 +101,7 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Add multiple sub elements to the base element
|
||||||
*
|
*
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $sub
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $sub
|
||||||
@@ -117,7 +117,7 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Remove all sub element entries
|
||||||
*
|
*
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
||||||
* @return array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>}
|
* @return array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>}
|
||||||
@@ -131,7 +131,7 @@ class Block
|
|||||||
// CSS Elements
|
// CSS Elements
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Add css entry to the css entries
|
||||||
*
|
*
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
||||||
* @param string ...$css
|
* @param string ...$css
|
||||||
@@ -144,7 +144,7 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Remove a css entry entry from the css array
|
||||||
*
|
*
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $element
|
||||||
* @param string ...$css
|
* @param string ...$css
|
||||||
@@ -157,7 +157,7 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Switch CSS entries
|
||||||
* scssel (switch) is not supported
|
* scssel (switch) is not supported
|
||||||
* use rcssel -> acssel
|
* use rcssel -> acssel
|
||||||
*
|
*
|
||||||
@@ -175,7 +175,7 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Build HTML from the content tree
|
||||||
* alias phfo
|
* alias phfo
|
||||||
*
|
*
|
||||||
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $tree
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $tree
|
||||||
@@ -231,7 +231,19 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Alias for phfo
|
||||||
|
*
|
||||||
|
* @param array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>} $tree
|
||||||
|
* @param bool $add_nl [default=false]
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public static function phfo(array $tree, bool $add_nl = false): string
|
||||||
|
{
|
||||||
|
return self::buildHtml($tree, $add_nl);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build HTML elements from an array of elements
|
||||||
* alias phfa
|
* alias phfa
|
||||||
*
|
*
|
||||||
* @param array<array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>}> $list
|
* @param array<array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>}> $list
|
||||||
@@ -248,8 +260,7 @@ class Block
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* alias for buildHtmlFromList
|
||||||
* wrapper for buildHtmlFromList
|
|
||||||
*
|
*
|
||||||
* @param array<array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>}> $list array of Elements to build string from
|
* @param array<array{tag:string,id:string,name:string,content:string,css:array<string>,options:array<string,string>,sub:array<mixed>}> $list array of Elements to build string from
|
||||||
* @param bool $add_nl [default=false] Optional output string line break
|
* @param bool $add_nl [default=false] Optional output string line break
|
||||||
|
|||||||
@@ -401,7 +401,7 @@ class Element
|
|||||||
* @param bool $add_nl [default=false] Optional output string line breaks
|
* @param bool $add_nl [default=false] Optional output string line breaks
|
||||||
* @return string HTML as string
|
* @return string HTML as string
|
||||||
*/
|
*/
|
||||||
public function buildHtml(Element $tree = null, bool $add_nl = false): string
|
public function buildHtml(?Element $tree = null, bool $add_nl = false): string
|
||||||
{
|
{
|
||||||
// print "D01: " . microtime(true) . "<br>";
|
// print "D01: " . microtime(true) . "<br>";
|
||||||
if ($tree === null) {
|
if ($tree === null) {
|
||||||
@@ -533,7 +533,7 @@ class Element
|
|||||||
* @return string build html as string
|
* @return string build html as string
|
||||||
* @deprecated Do not use, use Element->buildHtml() instead
|
* @deprecated Do not use, use Element->buildHtml() instead
|
||||||
*/
|
*/
|
||||||
public static function printHtmlFromObject(Element $tree = null, bool $add_nl = false): string
|
public static function printHtmlFromObject(?Element $tree = null, bool $add_nl = false): string
|
||||||
{
|
{
|
||||||
// nothing ->bad
|
// nothing ->bad
|
||||||
if ($tree === null) {
|
if ($tree === null) {
|
||||||
|
|||||||
@@ -185,7 +185,7 @@ class SmartyExtend extends \Smarty
|
|||||||
// call basic smarty
|
// call basic smarty
|
||||||
// or Smarty::__construct();
|
// or Smarty::__construct();
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
// iinit lang
|
// init lang
|
||||||
$this->l10n = $l10n;
|
$this->l10n = $l10n;
|
||||||
// parse and read, legacy stuff
|
// parse and read, legacy stuff
|
||||||
$locale = $this->l10n->getLocaleAsArray();
|
$locale = $this->l10n->getLocaleAsArray();
|
||||||
@@ -203,7 +203,8 @@ class SmartyExtend extends \Smarty
|
|||||||
_bind_textdomain_codeset($this->domain, $this->encoding);
|
_bind_textdomain_codeset($this->domain, $this->encoding);
|
||||||
|
|
||||||
// register smarty variable
|
// register smarty variable
|
||||||
$this->registerPlugin('modifier', 'getvar', [&$this, 'getTemplateVars']);
|
// $this->registerPlugin(\Smarty\Smarty::PLUGIN_MODIFIER, 'getvar', [&$this, 'getTemplateVars']);
|
||||||
|
$this->registerPlugin(self::PLUGIN_MODIFIER, 'getvar', [&$this, 'getTemplateVars']);
|
||||||
|
|
||||||
$this->page_name = \CoreLibs\Get\System::getPageName();
|
$this->page_name = \CoreLibs\Get\System::getPageName();
|
||||||
|
|
||||||
|
|||||||
1041
src/UrlRequests/Curl.php
Normal file
1041
src/UrlRequests/Curl.php
Normal file
File diff suppressed because it is too large
Load Diff
128
src/UrlRequests/CurlTrait.php
Normal file
128
src/UrlRequests/CurlTrait.php
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/10/29
|
||||||
|
* DESCRIPTION:
|
||||||
|
* Curl Client Trait for get/post/put/delete requests through the php curl inteface
|
||||||
|
*
|
||||||
|
* For anything more complex use guzzlehttp/http
|
||||||
|
* https://docs.guzzlephp.org/en/stable/index.html
|
||||||
|
*/
|
||||||
|
|
||||||
|
// phpcs:disable Generic.Files.LineLength
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace CoreLibs\UrlRequests;
|
||||||
|
|
||||||
|
trait CurlTrait
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* combined set call for any type of request with options type parameters
|
||||||
|
* The following options can be set:
|
||||||
|
* header: as array string:string
|
||||||
|
* query as string or array string:string
|
||||||
|
* body as string or array of any type
|
||||||
|
*
|
||||||
|
* @param string $type What type of request we send, will throw exception if not a valid one
|
||||||
|
* @param string $url The url to send
|
||||||
|
* @param array{auth?:null|array{0:string,1:string,2:string},headers?:null|array<string,string|array<string>>,query?:null|array<string,string>,body?:null|string|array<mixed>,http_errors?:null|bool} $options Request options
|
||||||
|
* @return array{code:string,headers:array<string,array<string>>,content:string} [default=[]] Result code, headers and content as array, content is json
|
||||||
|
* @throws \UnexpectedValueException on missing body data when body data is needed
|
||||||
|
*/
|
||||||
|
abstract public function request(string $type, string $url, array $options = []): array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Makes an request to the target url via curl: GET
|
||||||
|
* Returns result as string (json)
|
||||||
|
*
|
||||||
|
* @param string $url The URL being requested,
|
||||||
|
* including domain and protocol
|
||||||
|
* @param array{auth?:null|array{0:string,1:string,2:string},headers?:null|array<string,string|array<string>>,query?:null|array<string,string>,body?:null|string|array<mixed>,http_errors?:null|bool} $options Options to set
|
||||||
|
* @return array{code:string,headers:array<string,array<string>>,content:string} [default=[]] Result code, headers and content as array, content is json
|
||||||
|
*/
|
||||||
|
public function get(string $url, array $options = []): array
|
||||||
|
{
|
||||||
|
return $this->request(
|
||||||
|
"get",
|
||||||
|
$url,
|
||||||
|
$options,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Makes an request to the target url via curl: POST
|
||||||
|
* Returns result as string (json)
|
||||||
|
*
|
||||||
|
* @param string $url The URL being requested,
|
||||||
|
* including domain and protocol
|
||||||
|
* @param array{auth?:null|array{0:string,1:string,2:string},headers?:null|array<string,string|array<string>>,query?:null|array<string,string>,body?:null|string|array<mixed>,http_errors?:null|bool} $options Options to set
|
||||||
|
* @return array{code:string,headers:array<string,array<string>>,content:string} Result code, headers and content as array, content is json
|
||||||
|
*/
|
||||||
|
public function post(string $url, array $options): array
|
||||||
|
{
|
||||||
|
return $this->request(
|
||||||
|
"post",
|
||||||
|
$url,
|
||||||
|
$options,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Makes an request to the target url via curl: PUT
|
||||||
|
* Returns result as string (json)
|
||||||
|
*
|
||||||
|
* @param string $url The URL being requested,
|
||||||
|
* including domain and protocol
|
||||||
|
* @param array{auth?:null|array{0:string,1:string,2:string},headers?:null|array<string,string|array<string>>,query?:null|array<string,string>,body?:null|string|array<mixed>,http_errors?:null|bool} $options Options to set
|
||||||
|
* @return array{code:string,headers:array<string,array<string>>,content:string} Result code, headers and content as array, content is json
|
||||||
|
*/
|
||||||
|
public function put(string $url, array $options): array
|
||||||
|
{
|
||||||
|
return $this->request(
|
||||||
|
"put",
|
||||||
|
$url,
|
||||||
|
$options,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Makes an request to the target url via curl: PATCH
|
||||||
|
* Returns result as string (json)
|
||||||
|
*
|
||||||
|
* @param string $url The URL being requested,
|
||||||
|
* including domain and protocol
|
||||||
|
* @param array{auth?:null|array{0:string,1:string,2:string},headers?:null|array<string,string|array<string>>,query?:null|array<string,string>,body?:null|string|array<mixed>,http_errors?:null|bool} $options Options to set
|
||||||
|
* @return array{code:string,headers:array<string,array<string>>,content:string} Result code, headers and content as array, content is json
|
||||||
|
*/
|
||||||
|
public function patch(string $url, array $options): array
|
||||||
|
{
|
||||||
|
return $this->request(
|
||||||
|
"patch",
|
||||||
|
$url,
|
||||||
|
$options,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Makes an request to the target url via curl: DELETE
|
||||||
|
* Returns result as string (json)
|
||||||
|
* Note that DELETE body is optional
|
||||||
|
*
|
||||||
|
* @param string $url The URL being requested,
|
||||||
|
* including domain and protocol
|
||||||
|
* @param array{auth?:null|array{0:string,1:string,2:string},headers?:null|array<string,string|array<string>>,query?:null|array<string,string>,body?:null|string|array<mixed>,http_errors?:null|bool} $options Options to set
|
||||||
|
* @return array{code:string,headers:array<string,array<string>>,content:string} [default=[]] Result code, headers and content as array, content is json
|
||||||
|
*/
|
||||||
|
public function delete(string $url, array $options = []): array
|
||||||
|
{
|
||||||
|
return $this->request(
|
||||||
|
"delete",
|
||||||
|
$url,
|
||||||
|
$options,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
83
src/UrlRequests/Interface/RequestsInterface.php
Normal file
83
src/UrlRequests/Interface/RequestsInterface.php
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2024/9/20
|
||||||
|
* DESCRIPTION:
|
||||||
|
* URL Requests client interface
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace CoreLibs\UrlRequests\Interface;
|
||||||
|
|
||||||
|
interface RequestsInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* get the config array with all settings
|
||||||
|
*
|
||||||
|
* @return array<string,mixed> all current config settings
|
||||||
|
*/
|
||||||
|
public function getConfig(): array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the full url as it was sent
|
||||||
|
*
|
||||||
|
* @return string url sent
|
||||||
|
*/
|
||||||
|
public function getUrlSent(): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get the parsed url
|
||||||
|
*
|
||||||
|
* @return array{scheme?:string,user?:string,host?:string,port?:string,path?:string,query?:string,fragment?:string,pass?:string}
|
||||||
|
*/
|
||||||
|
public function getUrlParsedSent(): array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the full headers as they where sent
|
||||||
|
*
|
||||||
|
* @return array<string,string>
|
||||||
|
*/
|
||||||
|
public function getHeadersSent(): array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set, add or overwrite header
|
||||||
|
* On default this will overwrite header, and not set
|
||||||
|
*
|
||||||
|
* @param array<string,string|array<string>> $header
|
||||||
|
* @param bool $add [default=false] if set will add header to existing value
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function setHeaders(array $header, bool $add = false): void;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* remove header entry
|
||||||
|
* if key is only set then match only key, if both are set both sides must match
|
||||||
|
*
|
||||||
|
* @param array<string,string> $remove_headers
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function removeHeaders(array $remove_headers): void;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update the base url set, if empty will unset the base url
|
||||||
|
*
|
||||||
|
* @param string $base_uri
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function setBaseUri(string $base_uri): void;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* combined set call for any type of request with options type parameters
|
||||||
|
*
|
||||||
|
* phpcs:disable Generic.Files.LineLength
|
||||||
|
* @param string $type
|
||||||
|
* @param string $url
|
||||||
|
* @param array{auth?:null|array{0:string,1:string,2:string},headers?:null|array<string,string|array<string>>,query?:null|array<string,string>,body?:null|string|array<mixed>,http_errors?:null|bool} $options
|
||||||
|
* @return array{code:string,headers:array<string,array<string>>,content:string} Result code, headers and content as array, content is json
|
||||||
|
* @throws \UnexpectedValueException on missing body data when body data is needed
|
||||||
|
* phpcs:enable Generic.Files.LineLength
|
||||||
|
*/
|
||||||
|
public function request(string $type, string $url, array $options = []): array;
|
||||||
|
}
|
||||||
|
|
||||||
|
// __END__
|
||||||
@@ -1,2 +1,2 @@
|
|||||||
base="/storage/var/www/html/developers/clemens/core_data/composer-packages/CoreLibs-Composer-All/";
|
base="/storage/var/www/html/developers/clemens/core_data/composer-packages/CoreLibs-Composer-All/";
|
||||||
vendor/bin/phan --progress-bar -C --analyze-twice
|
tools/phan --progress-bar -C --analyze-twice
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
base="/storage/var/www/html/developers/clemens/core_data/composer-packages/CoreLibs-Composer-All/";
|
base="/storage/var/www/html/developers/clemens/core_data/composer-packages/CoreLibs-Composer-All/";
|
||||||
vendor/bin/phpstan
|
tools/phpstan
|
||||||
|
|||||||
@@ -15,25 +15,27 @@ php_bin="";
|
|||||||
if [ ! -z "${1}" ]; then
|
if [ ! -z "${1}" ]; then
|
||||||
case "${1}" in
|
case "${1}" in
|
||||||
# "7.3") php_bin="/usr/bin/php7.3 "; ;;
|
# "7.3") php_bin="/usr/bin/php7.3 "; ;;
|
||||||
"7.4") php_bin="/usr/bin/php7.4 "; ;;
|
#"7.4") php_bin="/usr/bin/php7.4 "; ;;
|
||||||
"8.0") php_bin="/usr/bin/php8.0 "; ;;
|
#"8.0") php_bin="/usr/bin/php8.0 "; ;;
|
||||||
"8.1") php_bin="/usr/bin/php8.1 "; ;;
|
#"8.1") php_bin="/usr/bin/php8.1 "; ;;
|
||||||
"8.2") php_bin="/usr/bin/php8.2 "; ;;
|
"8.2") php_bin="/usr/bin/php8.2 "; ;;
|
||||||
|
"8.3") php_bin="/usr/bin/php8.3 "; ;;
|
||||||
*) echo "Not support PHP: ${1}"; exit; ;;
|
*) echo "Not support PHP: ${1}"; exit; ;;
|
||||||
esac;
|
esac;
|
||||||
fi;
|
fi;
|
||||||
if [ ! -z "${2}" ] && [ -z "${php_bin}" ]; then
|
if [ ! -z "${2}" ] && [ -z "${php_bin}" ]; then
|
||||||
case "${2}" in
|
case "${2}" in
|
||||||
# "7.3") php_bin="/usr/bin/php7.3 "; ;;
|
# "7.3") php_bin="/usr/bin/php7.3 "; ;;
|
||||||
"7.4") php_bin="/usr/bin/php7.4 "; ;;
|
#"7.4") php_bin="/usr/bin/php7.4 "; ;;
|
||||||
"8.0") php_bin="/usr/bin/php8.0 "; ;;
|
#"8.0") php_bin="/usr/bin/php8.0 "; ;;
|
||||||
"8.1") php_bin="/usr/bin/php8.1 "; ;;
|
#"8.1") php_bin="/usr/bin/php8.1 "; ;;
|
||||||
"8.2") php_bin="/usr/bin/php8.2 "; ;;
|
"8.2") php_bin="/usr/bin/php8.2 "; ;;
|
||||||
|
"8.3") php_bin="/usr/bin/php8.3 "; ;;
|
||||||
*) echo "Not support PHP: ${1}"; exit; ;;
|
*) echo "Not support PHP: ${1}"; exit; ;;
|
||||||
esac;
|
esac;
|
||||||
fi;
|
fi;
|
||||||
|
|
||||||
phpunit_call="${php_bin}${base}vendor/bin/phpunit ${opt_testdox} -c ${base}phpunit.xml ${base}test/phpunit/";
|
phpunit_call="${php_bin}${base}tools/phpunit ${opt_testdox} -c ${base}phpunit.xml ${base}test/phpunit/";
|
||||||
|
|
||||||
${phpunit_call};
|
${phpunit_call};
|
||||||
|
|
||||||
|
|||||||
@@ -100,27 +100,6 @@ define('DEFAULT_ACL_LEVEL', 80);
|
|||||||
/************* LOGOUT ********************/
|
/************* LOGOUT ********************/
|
||||||
// logout target
|
// logout target
|
||||||
define('LOGOUT_TARGET', '');
|
define('LOGOUT_TARGET', '');
|
||||||
// password change allowed
|
|
||||||
define('PASSWORD_CHANGE', false);
|
|
||||||
define('PASSWORD_FORGOT', false);
|
|
||||||
// min/max password length
|
|
||||||
define('PASSWORD_MIN_LENGTH', 9);
|
|
||||||
define('PASSWORD_MAX_LENGTH', 255);
|
|
||||||
// defines allowed special characters
|
|
||||||
define('PASSWORD_SPECIAL_RANGE', '@$!%*?&');
|
|
||||||
// password must have upper case, lower case, number, special
|
|
||||||
// comment out for not mandatory
|
|
||||||
define('PASSWORD_LOWER', '(?=.*[a-z])');
|
|
||||||
define('PASSWORD_UPPER', '(?=.*[A-Z])');
|
|
||||||
define('PASSWORD_NUMBER', '(?=.*\d)');
|
|
||||||
define('PASSWORD_SPECIAL', "(?=.*[" . PASSWORD_SPECIAL_RANGE . "])");
|
|
||||||
// define full regex
|
|
||||||
define('PASSWORD_REGEX', "/^"
|
|
||||||
. (defined('PASSWORD_LOWER') ? PASSWORD_LOWER : '')
|
|
||||||
. (defined('PASSWORD_UPPER') ? PASSWORD_UPPER : '')
|
|
||||||
. (defined('PASSWORD_NUMBER') ? PASSWORD_NUMBER : '')
|
|
||||||
. (defined('PASSWORD_SPECIAL') ? PASSWORD_SPECIAL : '')
|
|
||||||
. "[A-Za-z\d" . PASSWORD_SPECIAL_RANGE . "]{" . PASSWORD_MIN_LENGTH . "," . PASSWORD_MAX_LENGTH . "}$/");
|
|
||||||
|
|
||||||
/************* AJAX / ACCESS *************/
|
/************* AJAX / ACCESS *************/
|
||||||
// ajax request type
|
// ajax request type
|
||||||
@@ -161,13 +140,6 @@ define('DEFAULT_LOCALE', 'en_US.UTF-8');
|
|||||||
// default web page encoding setting
|
// default web page encoding setting
|
||||||
define('DEFAULT_ENCODING', 'UTF-8');
|
define('DEFAULT_ENCODING', 'UTF-8');
|
||||||
|
|
||||||
/************* LOGGING *******************/
|
|
||||||
// below two can be defined here, but they should be
|
|
||||||
// defined in either the header file or the file itself
|
|
||||||
// as $LOG_FILE_ID which takes presence over LOG_FILE_ID
|
|
||||||
// see Basic class constructor
|
|
||||||
define('LOG_FILE_ID', BASE_NAME);
|
|
||||||
|
|
||||||
/************* QUEUE TABLE *************/
|
/************* QUEUE TABLE *************/
|
||||||
// if we have a dev/live system
|
// if we have a dev/live system
|
||||||
// set_live is a per page/per item
|
// set_live is a per page/per item
|
||||||
@@ -291,22 +263,4 @@ if (file_exists(BASE . CONFIGS . 'config.other.php')) {
|
|||||||
require BASE . CONFIGS . 'config.other.php';
|
require BASE . CONFIGS . 'config.other.php';
|
||||||
}
|
}
|
||||||
|
|
||||||
/************* DEBUG *******************/
|
|
||||||
// turn off debug if debug flag is OFF
|
|
||||||
if (defined('DEBUG') && DEBUG == false) {
|
|
||||||
$ECHO_ALL = false;
|
|
||||||
$DEBUG_ALL = false;
|
|
||||||
$PRINT_ALL = false;
|
|
||||||
$DB_DEBUG = false;
|
|
||||||
$ENABLE_ERROR_HANDLING = false;
|
|
||||||
$DEBUG_ALL_OVERRIDE = false;
|
|
||||||
} else {
|
|
||||||
$ECHO_ALL = false;
|
|
||||||
$DEBUG_ALL = true;
|
|
||||||
$PRINT_ALL = true;
|
|
||||||
$DB_DEBUG = true;
|
|
||||||
$ENABLE_ERROR_HANDLING = false;
|
|
||||||
$DEBUG_ALL_OVERRIDE = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
28
test/configs/config.path.php
Normal file
28
test/configs/config.path.php
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<?php // phpcs:ignore PSR1.Files.SideEffects
|
||||||
|
|
||||||
|
/********************************************************************
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: 2018/10/11
|
||||||
|
* SHORT DESCRIPTION:
|
||||||
|
* configuration file for core path settings
|
||||||
|
* CSV target paths, and other download access URLS or paths needed
|
||||||
|
* HISTORY:
|
||||||
|
*********************************************************************/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
// find trigger name "admin/" or "frontend/" in the getcwd() folder
|
||||||
|
$folder = '';
|
||||||
|
foreach (['admin', 'frontend'] as $_folder) {
|
||||||
|
if (strstr(getcwd() ?: '', DIRECTORY_SEPARATOR . $_folder)) {
|
||||||
|
$folder = $_folder;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// if content path is empty, fallback is default
|
||||||
|
if (empty($folder)) {
|
||||||
|
$folder = 'default';
|
||||||
|
}
|
||||||
|
define('CONTENT_PATH', $folder . DIRECTORY_SEPARATOR);
|
||||||
|
|
||||||
|
// __END__
|
||||||
@@ -53,19 +53,6 @@ for (
|
|||||||
\gullevek\dotEnv\DotEnv::readEnvFile(
|
\gullevek\dotEnv\DotEnv::readEnvFile(
|
||||||
$__DIR__PATH . $CONFIG_PATH_PREFIX . CONFIG_PATH
|
$__DIR__PATH . $CONFIG_PATH_PREFIX . CONFIG_PATH
|
||||||
);
|
);
|
||||||
// find trigger name "admin/" or "frontend/" in the getcwd() folder
|
|
||||||
$folder = '';
|
|
||||||
foreach (['admin', 'frontend'] as $_folder) {
|
|
||||||
if (strstr(getcwd() ?: '', DIRECTORY_SEPARATOR . $_folder)) {
|
|
||||||
$folder = $_folder;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// if content path is empty, fallback is default
|
|
||||||
if (empty($folder)) {
|
|
||||||
$folder = 'default';
|
|
||||||
}
|
|
||||||
define('CONTENT_PATH', $folder . DIRECTORY_SEPARATOR);
|
|
||||||
// load master config file that loads all other config files
|
// load master config file that loads all other config files
|
||||||
require $__DIR__PATH . $CONFIG_PATH_PREFIX . CONFIG_PATH . 'config.master.php';
|
require $__DIR__PATH . $CONFIG_PATH_PREFIX . CONFIG_PATH . 'config.master.php';
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -4,4 +4,7 @@ require "../vendor/autoload.php";
|
|||||||
|
|
||||||
print "Bytes: " . CoreLibs\Convert\Byte::humanReadableByteFormat(123414) . "<br>";
|
print "Bytes: " . CoreLibs\Convert\Byte::humanReadableByteFormat(123414) . "<br>";
|
||||||
|
|
||||||
|
$curl = new CoreLibs\UrlRequests\Curl();
|
||||||
|
print "Config: " . print_r($curl->getConfig(), true) . "<br>";
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
65
test/phpunit/AAASetupData/requests/http_requests.php
Normal file
65
test/phpunit/AAASetupData/requests/http_requests.php
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
<?php // phpcs:ignore PSR1.Files.SideEffects
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AUTHOR: Clemens Schwaighofer
|
||||||
|
* CREATED: Ymd
|
||||||
|
* DESCRIPTION:
|
||||||
|
* DescriptionHere
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* build return json
|
||||||
|
*
|
||||||
|
* @param array<string,mixed> $http_headers
|
||||||
|
* @param ?string $body
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function buildContent(array $http_headers, ?string $body): string
|
||||||
|
{
|
||||||
|
if (is_string($body) && !empty($body)) {
|
||||||
|
$_body = json_decode($body, true);
|
||||||
|
if (!is_array($_body)) {
|
||||||
|
$body = [$body];
|
||||||
|
} else {
|
||||||
|
$body = $_body;
|
||||||
|
}
|
||||||
|
} elseif (is_string($body)) {
|
||||||
|
$body = [];
|
||||||
|
}
|
||||||
|
return json_encode([
|
||||||
|
'HEADERS' => $http_headers,
|
||||||
|
"REQUEST_TYPE" => $_SERVER['REQUEST_METHOD'],
|
||||||
|
"PARAMS" => $_GET,
|
||||||
|
"BODY" => $body,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$http_headers = array_filter($_SERVER, function ($value, $key) {
|
||||||
|
if (str_starts_with($key, 'HTTP_')) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}, ARRAY_FILTER_USE_BOTH);
|
||||||
|
|
||||||
|
header("Content-Type: application/json; charset=UTF-8");
|
||||||
|
|
||||||
|
// if the header has Authorization and RunAuthTest then exit with 401
|
||||||
|
if (!empty($http_headers['HTTP_AUTHORIZATION']) && !empty($http_headers['HTTP_RUNAUTHTEST'])) {
|
||||||
|
header("HTTP/1.1 401 Unauthorized");
|
||||||
|
print buildContent($http_headers, '{"code": 401, "content": {"Error": "Not Authorized"}}');
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// if server request type is get set file_get to null -> no body
|
||||||
|
if ($_SERVER['REQUEST_METHOD'] == "GET") {
|
||||||
|
$file_get = null;
|
||||||
|
} elseif (($file_get = file_get_contents('php://input')) === false) {
|
||||||
|
header("HTTP/1.1 404 Not Found");
|
||||||
|
print buildContent($http_headers, '{"code": 404, "content": {"Error": "file_get_contents failed"}}');
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
print buildContent($http_headers, $file_get);
|
||||||
|
|
||||||
|
// __END__
|
||||||
@@ -243,6 +243,8 @@ final class CoreLibsACLLoginTest extends TestCase
|
|||||||
[],
|
[],
|
||||||
[
|
[
|
||||||
'EUID' => 1,
|
'EUID' => 1,
|
||||||
|
'ECUID' => 'abc',
|
||||||
|
'ECUUID' => '1233456-1234-1234-1234-123456789012',
|
||||||
],
|
],
|
||||||
2,
|
2,
|
||||||
[],
|
[],
|
||||||
@@ -260,6 +262,8 @@ final class CoreLibsACLLoginTest extends TestCase
|
|||||||
[],
|
[],
|
||||||
[
|
[
|
||||||
'EUID' => 1,
|
'EUID' => 1,
|
||||||
|
'ECUID' => 'abc',
|
||||||
|
'ECUUID' => '1233456-1234-1234-1234-123456789012',
|
||||||
'USER_NAME' => '',
|
'USER_NAME' => '',
|
||||||
'GROUP_NAME' => '',
|
'GROUP_NAME' => '',
|
||||||
'ADMIN' => 1,
|
'ADMIN' => 1,
|
||||||
@@ -1085,9 +1089,9 @@ final class CoreLibsACLLoginTest extends TestCase
|
|||||||
/** @var \CoreLibs\Create\Session&MockObject */
|
/** @var \CoreLibs\Create\Session&MockObject */
|
||||||
$session_mock = $this->createPartialMock(
|
$session_mock = $this->createPartialMock(
|
||||||
\CoreLibs\Create\Session::class,
|
\CoreLibs\Create\Session::class,
|
||||||
['startSession', 'checkActiveSession', 'sessionDestroy']
|
['getSessionId', 'checkActiveSession', 'sessionDestroy']
|
||||||
);
|
);
|
||||||
$session_mock->method('startSession')->willReturn('ACLLOGINTEST12');
|
$session_mock->method('getSessionId')->willReturn('ACLLOGINTEST12');
|
||||||
$session_mock->method('checkActiveSession')->willReturn(true);
|
$session_mock->method('checkActiveSession')->willReturn(true);
|
||||||
$session_mock->method('sessionDestroy')->will(
|
$session_mock->method('sessionDestroy')->will(
|
||||||
$this->returnCallback(function () {
|
$this->returnCallback(function () {
|
||||||
@@ -1788,9 +1792,9 @@ final class CoreLibsACLLoginTest extends TestCase
|
|||||||
/** @var \CoreLibs\Create\Session&MockObject */
|
/** @var \CoreLibs\Create\Session&MockObject */
|
||||||
$session_mock = $this->createPartialMock(
|
$session_mock = $this->createPartialMock(
|
||||||
\CoreLibs\Create\Session::class,
|
\CoreLibs\Create\Session::class,
|
||||||
['startSession', 'checkActiveSession', 'sessionDestroy']
|
['getSessionId', 'checkActiveSession', 'sessionDestroy']
|
||||||
);
|
);
|
||||||
$session_mock->method('startSession')->willReturn('ACLLOGINTEST34');
|
$session_mock->method('getSessionId')->willReturn('ACLLOGINTEST34');
|
||||||
$session_mock->method('checkActiveSession')->willReturn(true);
|
$session_mock->method('checkActiveSession')->willReturn(true);
|
||||||
$session_mock->method('sessionDestroy')->will(
|
$session_mock->method('sessionDestroy')->will(
|
||||||
$this->returnCallback(function () {
|
$this->returnCallback(function () {
|
||||||
@@ -1902,9 +1906,9 @@ final class CoreLibsACLLoginTest extends TestCase
|
|||||||
/** @var \CoreLibs\Create\Session&MockObject */
|
/** @var \CoreLibs\Create\Session&MockObject */
|
||||||
$session_mock = $this->createPartialMock(
|
$session_mock = $this->createPartialMock(
|
||||||
\CoreLibs\Create\Session::class,
|
\CoreLibs\Create\Session::class,
|
||||||
['startSession', 'checkActiveSession', 'sessionDestroy']
|
['getSessionId', 'checkActiveSession', 'sessionDestroy']
|
||||||
);
|
);
|
||||||
$session_mock->method('startSession')->willReturn('ACLLOGINTEST34');
|
$session_mock->method('getSessionId')->willReturn('ACLLOGINTEST34');
|
||||||
$session_mock->method('checkActiveSession')->willReturn(true);
|
$session_mock->method('checkActiveSession')->willReturn(true);
|
||||||
$session_mock->method('sessionDestroy')->will(
|
$session_mock->method('sessionDestroy')->will(
|
||||||
$this->returnCallback(function () {
|
$this->returnCallback(function () {
|
||||||
@@ -1990,9 +1994,9 @@ final class CoreLibsACLLoginTest extends TestCase
|
|||||||
/** @var \CoreLibs\Create\Session&MockObject */
|
/** @var \CoreLibs\Create\Session&MockObject */
|
||||||
$session_mock = $this->createPartialMock(
|
$session_mock = $this->createPartialMock(
|
||||||
\CoreLibs\Create\Session::class,
|
\CoreLibs\Create\Session::class,
|
||||||
['startSession', 'checkActiveSession', 'sessionDestroy']
|
['getSessionId', 'checkActiveSession', 'sessionDestroy']
|
||||||
);
|
);
|
||||||
$session_mock->method('startSession')->willReturn('ACLLOGINTEST34');
|
$session_mock->method('getSessionId')->willReturn('ACLLOGINTEST34');
|
||||||
$session_mock->method('checkActiveSession')->willReturn(true);
|
$session_mock->method('checkActiveSession')->willReturn(true);
|
||||||
$session_mock->method('sessionDestroy')->will(
|
$session_mock->method('sessionDestroy')->will(
|
||||||
$this->returnCallback(function () {
|
$this->returnCallback(function () {
|
||||||
@@ -2086,9 +2090,9 @@ final class CoreLibsACLLoginTest extends TestCase
|
|||||||
/** @var \CoreLibs\Create\Session&MockObject */
|
/** @var \CoreLibs\Create\Session&MockObject */
|
||||||
$session_mock = $this->createPartialMock(
|
$session_mock = $this->createPartialMock(
|
||||||
\CoreLibs\Create\Session::class,
|
\CoreLibs\Create\Session::class,
|
||||||
['startSession', 'checkActiveSession', 'sessionDestroy']
|
['getSessionId', 'checkActiveSession', 'sessionDestroy']
|
||||||
);
|
);
|
||||||
$session_mock->method('startSession')->willReturn('ACLLOGINTEST34');
|
$session_mock->method('getSessionId')->willReturn('ACLLOGINTEST34');
|
||||||
$session_mock->method('checkActiveSession')->willReturn(true);
|
$session_mock->method('checkActiveSession')->willReturn(true);
|
||||||
$session_mock->method('sessionDestroy')->will(
|
$session_mock->method('sessionDestroy')->will(
|
||||||
$this->returnCallback(function () {
|
$this->returnCallback(function () {
|
||||||
|
|||||||
@@ -5,15 +5,15 @@ CREATE FUNCTION random_string(randomLength int)
|
|||||||
RETURNS text AS
|
RETURNS text AS
|
||||||
$$
|
$$
|
||||||
SELECT array_to_string(
|
SELECT array_to_string(
|
||||||
ARRAY(
|
ARRAY(
|
||||||
SELECT substring(
|
SELECT substring(
|
||||||
'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789',
|
'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789',
|
||||||
trunc(random() * 62)::int + 1,
|
trunc(random() * 62)::int + 1,
|
||||||
1
|
1
|
||||||
)
|
)
|
||||||
FROM generate_series(1, randomLength) AS gs(x)
|
FROM generate_series(1, randomLength) AS gs(x)
|
||||||
),
|
),
|
||||||
''
|
''
|
||||||
)
|
)
|
||||||
$$
|
$$
|
||||||
LANGUAGE SQL
|
LANGUAGE SQL
|
||||||
@@ -27,15 +27,16 @@ CREATE OR REPLACE FUNCTION set_edit_generic()
|
|||||||
RETURNS TRIGGER AS
|
RETURNS TRIGGER AS
|
||||||
$$
|
$$
|
||||||
DECLARE
|
DECLARE
|
||||||
random_length INT = 12; -- that should be long enough
|
random_length INT = 12; -- that should be long enough
|
||||||
BEGIN
|
BEGIN
|
||||||
IF TG_OP = 'INSERT' THEN
|
IF TG_OP = 'INSERT' THEN
|
||||||
NEW.date_created := 'now';
|
NEW.date_created := 'now';
|
||||||
NEW.cuid := random_string(random_length);
|
NEW.cuid := random_string(random_length);
|
||||||
ELSIF TG_OP = 'UPDATE' THEN
|
NEW.cuuid := gen_random_uuid();
|
||||||
NEW.date_updated := 'now';
|
ELSIF TG_OP = 'UPDATE' THEN
|
||||||
END IF;
|
NEW.date_updated := 'now';
|
||||||
RETURN NEW;
|
END IF;
|
||||||
|
RETURN NEW;
|
||||||
END;
|
END;
|
||||||
$$
|
$$
|
||||||
LANGUAGE 'plpgsql';
|
LANGUAGE 'plpgsql';
|
||||||
@@ -46,29 +47,29 @@ LANGUAGE 'plpgsql';
|
|||||||
CREATE OR REPLACE FUNCTION set_edit_access_uid() RETURNS TRIGGER AS
|
CREATE OR REPLACE FUNCTION set_edit_access_uid() RETURNS TRIGGER AS
|
||||||
$$
|
$$
|
||||||
DECLARE
|
DECLARE
|
||||||
myrec RECORD;
|
myrec RECORD;
|
||||||
v_uid VARCHAR;
|
v_uid VARCHAR;
|
||||||
BEGIN
|
BEGIN
|
||||||
-- skip if NEW.name is not set
|
-- skip if NEW.name is not set
|
||||||
IF NEW.name IS NOT NULL AND NEW.name <> '' THEN
|
IF NEW.name IS NOT NULL AND NEW.name <> '' THEN
|
||||||
-- use NEW.name as base, remove all spaces
|
-- use NEW.name as base, remove all spaces
|
||||||
-- name data is already unique, so we do not need to worry about this here
|
-- name data is already unique, so we do not need to worry about this here
|
||||||
v_uid := REPLACE(NEW.name, ' ', '');
|
v_uid := REPLACE(NEW.name, ' ', '');
|
||||||
IF TG_OP = 'INSERT' THEN
|
IF TG_OP = 'INSERT' THEN
|
||||||
-- always set
|
-- always set
|
||||||
NEW.uid := v_uid;
|
NEW.uid := v_uid;
|
||||||
ELSIF TG_OP = 'UPDATE' THEN
|
ELSIF TG_OP = 'UPDATE' THEN
|
||||||
-- check if not set, then set
|
-- check if not set, then set
|
||||||
SELECT INTO myrec t.* FROM edit_access t WHERE edit_access_id = NEW.edit_access_id;
|
SELECT INTO myrec t.* FROM edit_access t WHERE edit_access_id = NEW.edit_access_id;
|
||||||
IF FOUND THEN
|
IF FOUND THEN
|
||||||
NEW.uid := v_uid;
|
NEW.uid := v_uid;
|
||||||
END IF;
|
END IF;
|
||||||
END IF;
|
END IF;
|
||||||
END IF;
|
END IF;
|
||||||
RETURN NEW;
|
RETURN NEW;
|
||||||
END;
|
END;
|
||||||
$$
|
$$
|
||||||
LANGUAGE 'plpgsql';
|
LANGUAGE 'plpgsql';
|
||||||
-- END: function/edit_access_set_uid.sql
|
-- END: function/edit_access_set_uid.sql
|
||||||
-- START: function/edit_group_set_uid.sql
|
-- START: function/edit_group_set_uid.sql
|
||||||
-- add uid add for edit_group table
|
-- add uid add for edit_group table
|
||||||
@@ -76,29 +77,29 @@ $$
|
|||||||
CREATE OR REPLACE FUNCTION set_edit_group_uid() RETURNS TRIGGER AS
|
CREATE OR REPLACE FUNCTION set_edit_group_uid() RETURNS TRIGGER AS
|
||||||
$$
|
$$
|
||||||
DECLARE
|
DECLARE
|
||||||
myrec RECORD;
|
myrec RECORD;
|
||||||
v_uid VARCHAR;
|
v_uid VARCHAR;
|
||||||
BEGIN
|
BEGIN
|
||||||
-- skip if NEW.name is not set
|
-- skip if NEW.name is not set
|
||||||
IF NEW.name IS NOT NULL AND NEW.name <> '' THEN
|
IF NEW.name IS NOT NULL AND NEW.name <> '' THEN
|
||||||
-- use NEW.name as base, remove all spaces
|
-- use NEW.name as base, remove all spaces
|
||||||
-- name data is already unique, so we do not need to worry about this here
|
-- name data is already unique, so we do not need to worry about this here
|
||||||
v_uid := REPLACE(NEW.name, ' ', '');
|
v_uid := REPLACE(NEW.name, ' ', '');
|
||||||
IF TG_OP = 'INSERT' THEN
|
IF TG_OP = 'INSERT' THEN
|
||||||
-- always set
|
-- always set
|
||||||
NEW.uid := v_uid;
|
NEW.uid := v_uid;
|
||||||
ELSIF TG_OP = 'UPDATE' THEN
|
ELSIF TG_OP = 'UPDATE' THEN
|
||||||
-- check if not set, then set
|
-- check if not set, then set
|
||||||
SELECT INTO myrec t.* FROM edit_group t WHERE edit_group_id = NEW.edit_group_id;
|
SELECT INTO myrec t.* FROM edit_group t WHERE edit_group_id = NEW.edit_group_id;
|
||||||
IF FOUND THEN
|
IF FOUND THEN
|
||||||
NEW.uid := v_uid;
|
NEW.uid := v_uid;
|
||||||
END IF;
|
END IF;
|
||||||
END IF;
|
END IF;
|
||||||
END IF;
|
END IF;
|
||||||
RETURN NEW;
|
RETURN NEW;
|
||||||
END;
|
END;
|
||||||
$$
|
$$
|
||||||
LANGUAGE 'plpgsql';
|
LANGUAGE 'plpgsql';
|
||||||
-- END: function/edit_group_set_uid.sql
|
-- END: function/edit_group_set_uid.sql
|
||||||
-- START: function/edit_log_partition_insert.sql
|
-- START: function/edit_log_partition_insert.sql
|
||||||
-- AUTHOR: Clemens Schwaighofer
|
-- AUTHOR: Clemens Schwaighofer
|
||||||
@@ -112,142 +113,142 @@ CREATE OR REPLACE FUNCTION edit_log_insert_trigger ()
|
|||||||
RETURNS TRIGGER AS
|
RETURNS TRIGGER AS
|
||||||
$$
|
$$
|
||||||
DECLARE
|
DECLARE
|
||||||
start_date DATE := '2010-01-01';
|
start_date DATE := '2010-01-01';
|
||||||
end_date DATE;
|
end_date DATE;
|
||||||
timeformat TEXT := 'YYYY';
|
timeformat TEXT := 'YYYY';
|
||||||
selector TEXT := 'year';
|
selector TEXT := 'year';
|
||||||
base_table TEXT := 'edit_log';
|
base_table TEXT := 'edit_log';
|
||||||
_interval INTERVAL := '1 ' || selector;
|
_interval INTERVAL := '1 ' || selector;
|
||||||
_interval_next INTERVAL := '2 ' || selector;
|
_interval_next INTERVAL := '2 ' || selector;
|
||||||
table_name TEXT;
|
table_name TEXT;
|
||||||
-- compare date column
|
-- compare date column
|
||||||
compare_date DATE := NEW.event_date;
|
compare_date DATE := NEW.event_date;
|
||||||
compare_date_name TEXT := 'event_date';
|
compare_date_name TEXT := 'event_date';
|
||||||
-- the create commands
|
-- the create commands
|
||||||
command_create_table TEXT := 'CREATE TABLE IF NOT EXISTS {TABLE_NAME} (CHECK({COMPARE_DATE_NAME} >= {START_DATE} AND {COMPARE_DATE_NAME} < {END_DATE})) INHERITS ({BASE_NAME})';
|
command_create_table TEXT := 'CREATE TABLE IF NOT EXISTS {TABLE_NAME} (CHECK({COMPARE_DATE_NAME} >= {START_DATE} AND {COMPARE_DATE_NAME} < {END_DATE})) INHERITS ({BASE_NAME})';
|
||||||
command_create_primary_key TEXT := 'ALTER TABLE {TABLE_NAME} ADD PRIMARY KEY ({BASE_TABLE}_id)';
|
command_create_primary_key TEXT := 'ALTER TABLE {TABLE_NAME} ADD PRIMARY KEY ({BASE_TABLE}_id)';
|
||||||
command_create_foreign_key_1 TEXT := 'ALTER TABLE {TABLE_NAME} ADD CONSTRAINT {TABLE_NAME}_euid_fkey FOREIGN KEY (euid) REFERENCES edit_user (edit_user_id) MATCH FULL ON UPDATE CASCADE ON DELETE SET NULL';
|
command_create_foreign_key_1 TEXT := 'ALTER TABLE {TABLE_NAME} ADD CONSTRAINT {TABLE_NAME}_euid_fkey FOREIGN KEY (euid) REFERENCES edit_user (edit_user_id) MATCH FULL ON UPDATE CASCADE ON DELETE SET NULL';
|
||||||
command_create_trigger_1 TEXT = 'CREATE TRIGGER trg_{TABLE_NAME} BEFORE INSERT OR UPDATE ON {TABLE_NAME} FOR EACH ROW EXECUTE PROCEDURE set_edit_generic()';
|
command_create_trigger_1 TEXT = 'CREATE TRIGGER trg_{TABLE_NAME} BEFORE INSERT OR UPDATE ON {TABLE_NAME} FOR EACH ROW EXECUTE PROCEDURE set_edit_generic()';
|
||||||
BEGIN
|
BEGIN
|
||||||
-- we are in valid start time area
|
-- we are in valid start time area
|
||||||
IF (NEW.event_date >= start_date) THEN
|
IF (NEW.event_date >= start_date) THEN
|
||||||
-- current table name
|
-- current table name
|
||||||
table_name := base_table || '_' || to_char(NEW.event_date, timeformat);
|
table_name := base_table || '_' || to_char(NEW.event_date, timeformat);
|
||||||
BEGIN
|
BEGIN
|
||||||
EXECUTE 'INSERT INTO ' || quote_ident(table_name) || ' SELECT ($1).*' USING NEW;
|
EXECUTE 'INSERT INTO ' || quote_ident(table_name) || ' SELECT ($1).*' USING NEW;
|
||||||
-- if insert failed because of missing table, create new below
|
-- if insert failed because of missing table, create new below
|
||||||
EXCEPTION
|
EXCEPTION
|
||||||
WHEN undefined_table THEN
|
WHEN undefined_table THEN
|
||||||
-- another block, so in case the creation fails here too
|
-- another block, so in case the creation fails here too
|
||||||
BEGIN
|
BEGIN
|
||||||
-- create new table here + all indexes
|
-- create new table here + all indexes
|
||||||
start_date := date_trunc(selector, NEW.event_date);
|
start_date := date_trunc(selector, NEW.event_date);
|
||||||
end_date := date_trunc(selector, NEW.event_date + _interval);
|
end_date := date_trunc(selector, NEW.event_date + _interval);
|
||||||
-- creat table
|
-- creat table
|
||||||
EXECUTE format(REPLACE( -- end date
|
EXECUTE format(REPLACE( -- end date
|
||||||
REPLACE( -- start date
|
REPLACE( -- start date
|
||||||
REPLACE( -- compare date name
|
REPLACE( -- compare date name
|
||||||
REPLACE( -- base name (inherit)
|
REPLACE( -- base name (inherit)
|
||||||
REPLACE( -- table name
|
REPLACE( -- table name
|
||||||
command_create_table,
|
command_create_table,
|
||||||
'{TABLE_NAME}',
|
'{TABLE_NAME}',
|
||||||
table_name
|
table_name
|
||||||
),
|
),
|
||||||
'{BASE_NAME}',
|
'{BASE_NAME}',
|
||||||
base_table
|
base_table
|
||||||
),
|
),
|
||||||
'{COMPARE_DATE_NAME}',
|
'{COMPARE_DATE_NAME}',
|
||||||
compare_date_name
|
compare_date_name
|
||||||
),
|
),
|
||||||
'{START_DATE}',
|
'{START_DATE}',
|
||||||
quote_literal(start_date)
|
quote_literal(start_date)
|
||||||
),
|
),
|
||||||
'{END_DATE}',
|
'{END_DATE}',
|
||||||
quote_literal(end_date)
|
quote_literal(end_date)
|
||||||
));
|
));
|
||||||
-- create all indexes and triggers
|
-- create all indexes and triggers
|
||||||
EXECUTE format(REPLACE(
|
EXECUTE format(REPLACE(
|
||||||
REPLACE(
|
REPLACE(
|
||||||
command_create_primary_key,
|
command_create_primary_key,
|
||||||
'{TABLE_NAME}',
|
'{TABLE_NAME}',
|
||||||
table_name
|
table_name
|
||||||
),
|
),
|
||||||
'{BASE_TABLE}',
|
'{BASE_TABLE}',
|
||||||
base_table
|
base_table
|
||||||
));
|
));
|
||||||
-- FK constraints
|
-- FK constraints
|
||||||
EXECUTE format(REPLACE(command_create_foreign_key_1, '{TABLE_NAME}', table_name));
|
EXECUTE format(REPLACE(command_create_foreign_key_1, '{TABLE_NAME}', table_name));
|
||||||
-- generic trigger
|
-- generic trigger
|
||||||
EXECUTE format(REPLACE(command_create_trigger_1, '{TABLE_NAME}', table_name));
|
EXECUTE format(REPLACE(command_create_trigger_1, '{TABLE_NAME}', table_name));
|
||||||
|
|
||||||
-- insert try again
|
-- insert try again
|
||||||
EXECUTE 'INSERT INTO ' || quote_ident(table_name) || ' SELECT ($1).*' USING NEW;
|
EXECUTE 'INSERT INTO ' || quote_ident(table_name) || ' SELECT ($1).*' USING NEW;
|
||||||
EXCEPTION
|
EXCEPTION
|
||||||
WHEN OTHERS THEN
|
WHEN OTHERS THEN
|
||||||
-- if this faled, throw it into the overflow table (so we don't loose anything)
|
-- if this faled, throw it into the overflow table (so we don't loose anything)
|
||||||
INSERT INTO edit_log_overflow VALUES (NEW.*);
|
INSERT INTO edit_log_overflow VALUES (NEW.*);
|
||||||
END;
|
END;
|
||||||
-- other errors, insert into overlow
|
-- other errors, insert into overlow
|
||||||
WHEN OTHERS THEN
|
WHEN OTHERS THEN
|
||||||
-- if this faled, throw it into the overflow table (so we don't loose anything)
|
-- if this faled, throw it into the overflow table (so we don't loose anything)
|
||||||
INSERT INTO edit_log_overflow VALUES (NEW.*);
|
INSERT INTO edit_log_overflow VALUES (NEW.*);
|
||||||
END;
|
END;
|
||||||
-- main insert run done, check if we have to create next months table
|
-- main insert run done, check if we have to create next months table
|
||||||
BEGIN
|
BEGIN
|
||||||
-- check if next month table exists
|
-- check if next month table exists
|
||||||
table_name := base_table || '_' || to_char((SELECT NEW.event_date + _interval)::DATE, timeformat);
|
table_name := base_table || '_' || to_char((SELECT NEW.event_date + _interval)::DATE, timeformat);
|
||||||
-- RAISE NOTICE 'SEARCH NEXT: %', table_name;
|
-- RAISE NOTICE 'SEARCH NEXT: %', table_name;
|
||||||
IF (SELECT to_regclass(table_name)) IS NULL THEN
|
IF (SELECT to_regclass(table_name)) IS NULL THEN
|
||||||
-- move inner interval same
|
-- move inner interval same
|
||||||
start_date := date_trunc(selector, NEW.event_date + _interval);
|
start_date := date_trunc(selector, NEW.event_date + _interval);
|
||||||
end_date := date_trunc(selector, NEW.event_date + _interval_next);
|
end_date := date_trunc(selector, NEW.event_date + _interval_next);
|
||||||
-- RAISE NOTICE 'CREATE NEXT: %', table_name;
|
-- RAISE NOTICE 'CREATE NEXT: %', table_name;
|
||||||
-- create table
|
-- create table
|
||||||
EXECUTE format(REPLACE( -- end date
|
EXECUTE format(REPLACE( -- end date
|
||||||
REPLACE( -- start date
|
REPLACE( -- start date
|
||||||
REPLACE( -- compare date name
|
REPLACE( -- compare date name
|
||||||
REPLACE( -- base name (inherit)
|
REPLACE( -- base name (inherit)
|
||||||
REPLACE( -- table name
|
REPLACE( -- table name
|
||||||
command_create_table,
|
command_create_table,
|
||||||
'{TABLE_NAME}',
|
'{TABLE_NAME}',
|
||||||
table_name
|
table_name
|
||||||
),
|
),
|
||||||
'{BASE_NAME}',
|
'{BASE_NAME}',
|
||||||
base_table
|
base_table
|
||||||
),
|
),
|
||||||
'{COMPARE_DATE_NAME}',
|
'{COMPARE_DATE_NAME}',
|
||||||
compare_date_name
|
compare_date_name
|
||||||
),
|
),
|
||||||
'{START_DATE}',
|
'{START_DATE}',
|
||||||
quote_literal(start_date)
|
quote_literal(start_date)
|
||||||
),
|
),
|
||||||
'{END_DATE}',
|
'{END_DATE}',
|
||||||
quote_literal(end_date)
|
quote_literal(end_date)
|
||||||
));
|
));
|
||||||
-- create all indexes and triggers
|
-- create all indexes and triggers
|
||||||
EXECUTE format(REPLACE(
|
EXECUTE format(REPLACE(
|
||||||
REPLACE(
|
REPLACE(
|
||||||
command_create_primary_key,
|
command_create_primary_key,
|
||||||
'{TABLE_NAME}',
|
'{TABLE_NAME}',
|
||||||
table_name
|
table_name
|
||||||
),
|
),
|
||||||
'{BASE_TABLE}',
|
'{BASE_TABLE}',
|
||||||
base_table
|
base_table
|
||||||
));
|
));
|
||||||
-- FK constraints
|
-- FK constraints
|
||||||
EXECUTE format(REPLACE(command_create_foreign_key_1, '{TABLE_NAME}', table_name));
|
EXECUTE format(REPLACE(command_create_foreign_key_1, '{TABLE_NAME}', table_name));
|
||||||
-- generic trigger
|
-- generic trigger
|
||||||
EXECUTE format(REPLACE(command_create_trigger_1, '{TABLE_NAME}', table_name));
|
EXECUTE format(REPLACE(command_create_trigger_1, '{TABLE_NAME}', table_name));
|
||||||
END IF;
|
END IF;
|
||||||
EXCEPTION
|
EXCEPTION
|
||||||
WHEN OTHERS THEN
|
WHEN OTHERS THEN
|
||||||
RAISE NOTICE 'Failed to create next table: %', table_name;
|
RAISE NOTICE 'Failed to create next table: %', table_name;
|
||||||
END;
|
END;
|
||||||
ELSE
|
ELSE
|
||||||
-- if outside valid date, insert into overflow
|
-- if outside valid date, insert into overflow
|
||||||
INSERT INTO edit_log_overflow VALUES (NEW.*);
|
INSERT INTO edit_log_overflow VALUES (NEW.*);
|
||||||
END IF;
|
END IF;
|
||||||
RETURN NULL;
|
RETURN NULL;
|
||||||
END
|
END
|
||||||
$$
|
$$
|
||||||
LANGUAGE 'plpgsql';
|
LANGUAGE 'plpgsql';
|
||||||
@@ -260,22 +261,22 @@ CREATE OR REPLACE FUNCTION set_login_user_id_set_date()
|
|||||||
RETURNS TRIGGER AS
|
RETURNS TRIGGER AS
|
||||||
$$
|
$$
|
||||||
BEGIN
|
BEGIN
|
||||||
-- if new is not null/empty
|
-- if new is not null/empty
|
||||||
-- and old one is null or old one different new one
|
-- and old one is null or old one different new one
|
||||||
-- set NOW()
|
-- set NOW()
|
||||||
-- if new one is NULL
|
-- if new one is NULL
|
||||||
-- set NULL
|
-- set NULL
|
||||||
IF
|
IF
|
||||||
NEW.login_user_id IS NOT NULL AND NEW.login_user_id <> '' AND
|
NEW.login_user_id IS NOT NULL AND NEW.login_user_id <> '' AND
|
||||||
(OLD.login_user_id IS NULL OR NEW.login_user_id <> OLD.login_user_id)
|
(OLD.login_user_id IS NULL OR NEW.login_user_id <> OLD.login_user_id)
|
||||||
THEN
|
THEN
|
||||||
NEW.login_user_id_set_date = NOW();
|
NEW.login_user_id_set_date = NOW();
|
||||||
NEW.login_user_id_last_revalidate = NOW();
|
NEW.login_user_id_last_revalidate = NOW();
|
||||||
ELSIF NEW.login_user_id IS NULL OR NEW.login_user_id = '' THEN
|
ELSIF NEW.login_user_id IS NULL OR NEW.login_user_id = '' THEN
|
||||||
NEW.login_user_id_set_date = NULL;
|
NEW.login_user_id_set_date = NULL;
|
||||||
NEW.login_user_id_last_revalidate = NULL;
|
NEW.login_user_id_last_revalidate = NULL;
|
||||||
END IF;
|
END IF;
|
||||||
RETURN NEW;
|
RETURN NEW;
|
||||||
END;
|
END;
|
||||||
$$
|
$$
|
||||||
LANGUAGE 'plpgsql';
|
LANGUAGE 'plpgsql';
|
||||||
@@ -290,8 +291,8 @@ LANGUAGE 'plpgsql';
|
|||||||
|
|
||||||
-- DROP TABLE temp_files;
|
-- DROP TABLE temp_files;
|
||||||
CREATE TABLE temp_files (
|
CREATE TABLE temp_files (
|
||||||
filename VARCHAR,
|
filename VARCHAR,
|
||||||
folder VARCHAR
|
folder VARCHAR
|
||||||
);
|
);
|
||||||
-- END: table/edit_temp_files.sql
|
-- END: table/edit_temp_files.sql
|
||||||
-- START: table/edit_generic.sql
|
-- START: table/edit_generic.sql
|
||||||
@@ -304,9 +305,10 @@ CREATE TABLE temp_files (
|
|||||||
|
|
||||||
-- DROP TABLE edit_generic;
|
-- DROP TABLE edit_generic;
|
||||||
CREATE TABLE edit_generic (
|
CREATE TABLE edit_generic (
|
||||||
cuid VARCHAR,
|
cuid VARCHAR,
|
||||||
date_created TIMESTAMP WITHOUT TIME ZONE DEFAULT clock_timestamp(),
|
cuuid UUID,
|
||||||
date_updated TIMESTAMP WITHOUT TIME ZONE
|
date_created TIMESTAMP WITHOUT TIME ZONE DEFAULT clock_timestamp(),
|
||||||
|
date_updated TIMESTAMP WITHOUT TIME ZONE
|
||||||
);
|
);
|
||||||
-- END: table/edit_generic.sql
|
-- END: table/edit_generic.sql
|
||||||
-- START: table/edit_visible_group.sql
|
-- START: table/edit_visible_group.sql
|
||||||
@@ -319,9 +321,9 @@ CREATE TABLE edit_generic (
|
|||||||
|
|
||||||
-- DROP TABLE edit_visible_group;
|
-- DROP TABLE edit_visible_group;
|
||||||
CREATE TABLE edit_visible_group (
|
CREATE TABLE edit_visible_group (
|
||||||
edit_visible_group_id SERIAL PRIMARY KEY,
|
edit_visible_group_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
flag VARCHAR
|
flag VARCHAR
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_visible_group.sql
|
-- END: table/edit_visible_group.sql
|
||||||
-- START: table/edit_menu_group.sql
|
-- START: table/edit_menu_group.sql
|
||||||
@@ -334,10 +336,10 @@ CREATE TABLE edit_visible_group (
|
|||||||
|
|
||||||
-- DROP TABLE edit_menu_group;
|
-- DROP TABLE edit_menu_group;
|
||||||
CREATE TABLE edit_menu_group (
|
CREATE TABLE edit_menu_group (
|
||||||
edit_menu_group_id SERIAL PRIMARY KEY,
|
edit_menu_group_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
flag VARCHAR,
|
flag VARCHAR,
|
||||||
order_number INT NOT NULL
|
order_number INT NOT NULL
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
|
|
||||||
|
|
||||||
@@ -352,18 +354,18 @@ CREATE TABLE edit_menu_group (
|
|||||||
|
|
||||||
-- DROP TABLE edit_page;
|
-- DROP TABLE edit_page;
|
||||||
CREATE TABLE edit_page (
|
CREATE TABLE edit_page (
|
||||||
edit_page_id SERIAL PRIMARY KEY,
|
edit_page_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
content_alias_edit_page_id INT, -- alias for page content, if the page content is defined on a different page, ege for ajax backend pages
|
content_alias_edit_page_id INT, -- alias for page content, if the page content is defined on a different page, ege for ajax backend pages
|
||||||
FOREIGN KEY (content_alias_edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE RESTRICT ON UPDATE CASCADE,
|
FOREIGN KEY (content_alias_edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE RESTRICT ON UPDATE CASCADE,
|
||||||
filename VARCHAR,
|
filename VARCHAR,
|
||||||
name VARCHAR UNIQUE,
|
name VARCHAR UNIQUE,
|
||||||
order_number INT NOT NULL,
|
order_number INT NOT NULL,
|
||||||
online SMALLINT NOT NULL DEFAULT 0,
|
online SMALLINT NOT NULL DEFAULT 0,
|
||||||
menu SMALLINT NOT NULL DEFAULT 0,
|
menu SMALLINT NOT NULL DEFAULT 0,
|
||||||
popup SMALLINT NOT NULL DEFAULT 0,
|
popup SMALLINT NOT NULL DEFAULT 0,
|
||||||
popup_x SMALLINT,
|
popup_x SMALLINT,
|
||||||
popup_y SMALLINT,
|
popup_y SMALLINT,
|
||||||
hostname VARCHAR
|
hostname VARCHAR
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_page.sql
|
-- END: table/edit_page.sql
|
||||||
-- START: table/edit_query_string.sql
|
-- START: table/edit_query_string.sql
|
||||||
@@ -376,13 +378,13 @@ CREATE TABLE edit_page (
|
|||||||
|
|
||||||
-- DROP TABLE edit_query_string;
|
-- DROP TABLE edit_query_string;
|
||||||
CREATE TABLE edit_query_string (
|
CREATE TABLE edit_query_string (
|
||||||
edit_query_string_id SERIAL PRIMARY KEY,
|
edit_query_string_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
edit_page_id INT NOT NULL,
|
edit_page_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0,
|
enabled SMALLINT NOT NULL DEFAULT 0,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
value VARCHAR,
|
value VARCHAR,
|
||||||
dynamic SMALLINT NOT NULL DEFAULT 0
|
dynamic SMALLINT NOT NULL DEFAULT 0
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_query_string.sql
|
-- END: table/edit_query_string.sql
|
||||||
-- START: table/edit_page_visible_group.sql
|
-- START: table/edit_page_visible_group.sql
|
||||||
@@ -395,10 +397,10 @@ CREATE TABLE edit_query_string (
|
|||||||
|
|
||||||
-- DROP TABLE edit_page_visible_group;
|
-- DROP TABLE edit_page_visible_group;
|
||||||
CREATE TABLE edit_page_visible_group (
|
CREATE TABLE edit_page_visible_group (
|
||||||
edit_page_id INT NOT NULL,
|
edit_page_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_visible_group_id INT NOT NULL,
|
edit_visible_group_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_visible_group_id) REFERENCES edit_visible_group (edit_visible_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE
|
FOREIGN KEY (edit_visible_group_id) REFERENCES edit_visible_group (edit_visible_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE
|
||||||
);
|
);
|
||||||
-- END: table/edit_page_visible_group.sql
|
-- END: table/edit_page_visible_group.sql
|
||||||
-- START: table/edit_page_menu_group.sql
|
-- START: table/edit_page_menu_group.sql
|
||||||
@@ -411,10 +413,10 @@ CREATE TABLE edit_page_visible_group (
|
|||||||
|
|
||||||
-- DROP TABLE edit_page_menu_group;
|
-- DROP TABLE edit_page_menu_group;
|
||||||
CREATE TABLE edit_page_menu_group (
|
CREATE TABLE edit_page_menu_group (
|
||||||
edit_page_id INT NOT NULL,
|
edit_page_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_menu_group_id INT NOT NULL,
|
edit_menu_group_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_menu_group_id) REFERENCES edit_menu_group (edit_menu_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE
|
FOREIGN KEY (edit_menu_group_id) REFERENCES edit_menu_group (edit_menu_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE
|
||||||
);
|
);
|
||||||
-- END: table/edit_page_menu_group.sql
|
-- END: table/edit_page_menu_group.sql
|
||||||
-- START: table/edit_access_right.sql
|
-- START: table/edit_access_right.sql
|
||||||
@@ -428,11 +430,11 @@ CREATE TABLE edit_page_menu_group (
|
|||||||
|
|
||||||
-- DROP TABLE edit_access_right;
|
-- DROP TABLE edit_access_right;
|
||||||
CREATE TABLE edit_access_right (
|
CREATE TABLE edit_access_right (
|
||||||
edit_access_right_id SERIAL PRIMARY KEY,
|
edit_access_right_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
level SMALLINT,
|
level SMALLINT,
|
||||||
type VARCHAR,
|
type VARCHAR,
|
||||||
UNIQUE (level,type)
|
UNIQUE (level,type)
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_access_right.sql
|
-- END: table/edit_access_right.sql
|
||||||
-- START: table/edit_scheme.sql
|
-- START: table/edit_scheme.sql
|
||||||
@@ -445,12 +447,12 @@ CREATE TABLE edit_access_right (
|
|||||||
|
|
||||||
-- DROP TABLE edit_scheme;
|
-- DROP TABLE edit_scheme;
|
||||||
CREATE TABLE edit_scheme (
|
CREATE TABLE edit_scheme (
|
||||||
edit_scheme_id SERIAL PRIMARY KEY,
|
edit_scheme_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0,
|
enabled SMALLINT NOT NULL DEFAULT 0,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
header_color VARCHAR,
|
header_color VARCHAR,
|
||||||
css_file VARCHAR,
|
css_file VARCHAR,
|
||||||
template VARCHAR
|
template VARCHAR
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_scheme.sql
|
-- END: table/edit_scheme.sql
|
||||||
-- START: table/edit_language.sql
|
-- START: table/edit_language.sql
|
||||||
@@ -464,13 +466,13 @@ CREATE TABLE edit_scheme (
|
|||||||
|
|
||||||
-- DROP TABLE edit_language;
|
-- DROP TABLE edit_language;
|
||||||
CREATE TABLE edit_language (
|
CREATE TABLE edit_language (
|
||||||
edit_language_id SERIAL PRIMARY KEY,
|
edit_language_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0,
|
enabled SMALLINT NOT NULL DEFAULT 0,
|
||||||
lang_default SMALLINT NOT NULL DEFAULT 0,
|
lang_default SMALLINT NOT NULL DEFAULT 0,
|
||||||
long_name VARCHAR,
|
long_name VARCHAR,
|
||||||
short_name VARCHAR, -- en_US, en or en_US@latin without encoding
|
short_name VARCHAR, -- en_US, en or en_US@latin without encoding
|
||||||
iso_name VARCHAR, -- should actually be encoding
|
iso_name VARCHAR, -- should actually be encoding
|
||||||
order_number INT
|
order_number INT
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_language.sql
|
-- END: table/edit_language.sql
|
||||||
-- START: table/edit_group.sql
|
-- START: table/edit_group.sql
|
||||||
@@ -483,16 +485,16 @@ CREATE TABLE edit_language (
|
|||||||
|
|
||||||
-- DROP TABLE edit_group;
|
-- DROP TABLE edit_group;
|
||||||
CREATE TABLE edit_group (
|
CREATE TABLE edit_group (
|
||||||
edit_group_id SERIAL PRIMARY KEY,
|
edit_group_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
edit_scheme_id INT,
|
edit_scheme_id INT,
|
||||||
FOREIGN KEY (edit_scheme_id) REFERENCES edit_scheme (edit_scheme_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_scheme_id) REFERENCES edit_scheme (edit_scheme_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_access_right_id INT NOT NULL,
|
edit_access_right_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0,
|
enabled SMALLINT NOT NULL DEFAULT 0,
|
||||||
deleted SMALLINT DEFAULT 0,
|
deleted SMALLINT DEFAULT 0,
|
||||||
uid VARCHAR,
|
uid VARCHAR,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
additional_acl JSONB
|
additional_acl JSONB
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_group.sql
|
-- END: table/edit_group.sql
|
||||||
-- START: table/edit_page_access.sql
|
-- START: table/edit_page_access.sql
|
||||||
@@ -505,14 +507,14 @@ CREATE TABLE edit_group (
|
|||||||
|
|
||||||
-- DROP TABLE edit_page_access;
|
-- DROP TABLE edit_page_access;
|
||||||
CREATE TABLE edit_page_access (
|
CREATE TABLE edit_page_access (
|
||||||
edit_page_access_id SERIAL PRIMARY KEY,
|
edit_page_access_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
edit_group_id INT NOT NULL,
|
edit_group_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_group_id) REFERENCES edit_group (edit_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_group_id) REFERENCES edit_group (edit_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_page_id INT NOT NULL,
|
edit_page_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_access_right_id INT NOT NULL,
|
edit_access_right_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0
|
enabled SMALLINT NOT NULL DEFAULT 0
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
|
|
||||||
|
|
||||||
@@ -528,15 +530,15 @@ CREATE TABLE edit_page_access (
|
|||||||
|
|
||||||
-- DROP TABLE edit_page_content;
|
-- DROP TABLE edit_page_content;
|
||||||
CREATE TABLE edit_page_content (
|
CREATE TABLE edit_page_content (
|
||||||
edit_page_content_id SERIAL PRIMARY KEY,
|
edit_page_content_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
edit_page_id INT NOT NULL,
|
edit_page_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_page_id) REFERENCES edit_page (edit_page_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_access_right_id INT NOT NULL,
|
edit_access_right_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
uid VARCHAR UNIQUE,
|
uid VARCHAR UNIQUE,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
order_number INT NOT NULL,
|
order_number INT NOT NULL,
|
||||||
online SMALLINT NOT NULL DEFAULT 0
|
online SMALLINT NOT NULL DEFAULT 0
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_page_content.sql
|
-- END: table/edit_page_content.sql
|
||||||
-- START: table/edit_user.sql
|
-- START: table/edit_user.sql
|
||||||
@@ -549,63 +551,63 @@ CREATE TABLE edit_page_content (
|
|||||||
|
|
||||||
-- DROP TABLE edit_user;
|
-- DROP TABLE edit_user;
|
||||||
CREATE TABLE edit_user (
|
CREATE TABLE edit_user (
|
||||||
edit_user_id SERIAL PRIMARY KEY,
|
edit_user_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
connect_edit_user_id INT, -- possible reference to other user
|
connect_edit_user_id INT, -- possible reference to other user
|
||||||
FOREIGN KEY (connect_edit_user_id) REFERENCES edit_user (edit_user_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (connect_edit_user_id) REFERENCES edit_user (edit_user_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_language_id INT NOT NULL,
|
edit_language_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_language_id) REFERENCES edit_language (edit_language_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_language_id) REFERENCES edit_language (edit_language_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_group_id INT NOT NULL,
|
edit_group_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_group_id) REFERENCES edit_group (edit_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_group_id) REFERENCES edit_group (edit_group_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_scheme_id INT,
|
edit_scheme_id INT,
|
||||||
FOREIGN KEY (edit_scheme_id) REFERENCES edit_scheme (edit_scheme_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_scheme_id) REFERENCES edit_scheme (edit_scheme_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_access_right_id INT NOT NULL,
|
edit_access_right_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
-- username/password
|
-- username/password
|
||||||
username VARCHAR UNIQUE,
|
username VARCHAR UNIQUE,
|
||||||
password VARCHAR,
|
password VARCHAR,
|
||||||
-- name block
|
-- name block
|
||||||
first_name VARCHAR,
|
first_name VARCHAR,
|
||||||
last_name VARCHAR,
|
last_name VARCHAR,
|
||||||
first_name_furigana VARCHAR,
|
first_name_furigana VARCHAR,
|
||||||
last_name_furigana VARCHAR,
|
last_name_furigana VARCHAR,
|
||||||
-- email
|
-- email
|
||||||
email VARCHAR,
|
email VARCHAR,
|
||||||
-- eanbled/deleted flag
|
-- eanbled/deleted flag
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0,
|
enabled SMALLINT NOT NULL DEFAULT 0,
|
||||||
deleted SMALLINT NOT NULL DEFAULT 0,
|
deleted SMALLINT NOT NULL DEFAULT 0,
|
||||||
-- general flags
|
-- general flags
|
||||||
strict SMALLINT DEFAULT 0,
|
strict SMALLINT DEFAULT 0,
|
||||||
locked SMALLINT DEFAULT 0,
|
locked SMALLINT DEFAULT 0,
|
||||||
protected SMALLINT NOT NULL DEFAULT 0,
|
protected SMALLINT NOT NULL DEFAULT 0,
|
||||||
-- legacy, debug flags
|
-- legacy, debug flags
|
||||||
debug SMALLINT NOT NULL DEFAULT 0,
|
debug SMALLINT NOT NULL DEFAULT 0,
|
||||||
db_debug SMALLINT NOT NULL DEFAULT 0,
|
db_debug SMALLINT NOT NULL DEFAULT 0,
|
||||||
-- is admin user
|
-- is admin user
|
||||||
admin SMALLINT NOT NULL DEFAULT 0,
|
admin SMALLINT NOT NULL DEFAULT 0,
|
||||||
-- last login log
|
-- last login log
|
||||||
last_login TIMESTAMP WITHOUT TIME ZONE,
|
last_login TIMESTAMP WITHOUT TIME ZONE,
|
||||||
-- login error
|
-- login error
|
||||||
login_error_count INT DEFAULT 0,
|
login_error_count INT DEFAULT 0,
|
||||||
login_error_date_last TIMESTAMP WITHOUT TIME ZONE,
|
login_error_date_last TIMESTAMP WITHOUT TIME ZONE,
|
||||||
login_error_date_first TIMESTAMP WITHOUT TIME ZONE,
|
login_error_date_first TIMESTAMP WITHOUT TIME ZONE,
|
||||||
-- time locked
|
-- time locked
|
||||||
lock_until TIMESTAMP WITHOUT TIME ZONE,
|
lock_until TIMESTAMP WITHOUT TIME ZONE,
|
||||||
lock_after TIMESTAMP WITHOUT TIME ZONE,
|
lock_after TIMESTAMP WITHOUT TIME ZONE,
|
||||||
-- password change
|
-- password change
|
||||||
password_change_date TIMESTAMP WITHOUT TIME ZONE, -- only when password is first set or changed
|
password_change_date TIMESTAMP WITHOUT TIME ZONE, -- only when password is first set or changed
|
||||||
password_change_interval INTERVAL, -- null if no change is needed, or d/m/y time interval
|
password_change_interval INTERVAL, -- null if no change is needed, or d/m/y time interval
|
||||||
password_reset_time TIMESTAMP WITHOUT TIME ZONE, -- when the password reset was requested
|
password_reset_time TIMESTAMP WITHOUT TIME ZONE, -- when the password reset was requested
|
||||||
password_reset_uid VARCHAR, -- the uid to access the password reset page
|
password_reset_uid VARCHAR, -- the uid to access the password reset page
|
||||||
-- _GET login id for direct login
|
-- _GET login id for direct login
|
||||||
login_user_id VARCHAR UNIQUE, -- the loginUserId, at least 32 chars
|
login_user_id VARCHAR UNIQUE, -- the loginUserId, at least 32 chars
|
||||||
login_user_id_set_date TIMESTAMP WITHOUT TIME ZONE, -- when above uid was set
|
login_user_id_set_date TIMESTAMP WITHOUT TIME ZONE, -- when above uid was set
|
||||||
login_user_id_last_revalidate TIMESTAMP WITHOUT TIME ZONE, -- when the last login was done with user name and password
|
login_user_id_last_revalidate TIMESTAMP WITHOUT TIME ZONE, -- when the last login was done with user name and password
|
||||||
login_user_id_valid_from TIMESTAMP WITHOUT TIME ZONE, -- if set, from when the above uid is valid
|
login_user_id_valid_from TIMESTAMP WITHOUT TIME ZONE, -- if set, from when the above uid is valid
|
||||||
login_user_id_valid_until TIMESTAMP WITHOUT TIME ZONE, -- if set, until when the above uid is valid
|
login_user_id_valid_until TIMESTAMP WITHOUT TIME ZONE, -- if set, until when the above uid is valid
|
||||||
login_user_id_revalidate_after INTERVAL, -- user must login to revalidated loginUserId after set days, 0 for forever
|
login_user_id_revalidate_after INTERVAL, -- user must login to revalidated loginUserId after set days, 0 for forever
|
||||||
login_user_id_locked SMALLINT DEFAULT 0, -- lock for loginUserId, but still allow normal login
|
login_user_id_locked SMALLINT DEFAULT 0, -- lock for loginUserId, but still allow normal login
|
||||||
-- additional ACL json block
|
-- additional ACL json block
|
||||||
additional_acl JSONB -- additional ACL as JSON string (can be set by other pages)
|
additional_acl JSONB -- additional ACL as JSON string (can be set by other pages)
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
|
|
||||||
-- create unique index
|
-- create unique index
|
||||||
@@ -650,37 +652,40 @@ COMMENT ON COLUMN edit_user.additional_acl IS 'Additional Access Control List st
|
|||||||
|
|
||||||
-- DROP TABLE edit_log;
|
-- DROP TABLE edit_log;
|
||||||
CREATE TABLE edit_log (
|
CREATE TABLE edit_log (
|
||||||
edit_log_id SERIAL PRIMARY KEY,
|
edit_log_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
euid INT, -- this is a foreign key, but I don't nedd to reference to it
|
euid INT, -- this is a foreign key, but I don't nedd to reference to it
|
||||||
FOREIGN KEY (euid) REFERENCES edit_user (edit_user_id) MATCH FULL ON UPDATE CASCADE ON DELETE SET NULL,
|
FOREIGN KEY (euid) REFERENCES edit_user (edit_user_id) MATCH FULL ON UPDATE CASCADE ON DELETE SET NULL,
|
||||||
username VARCHAR,
|
ecuid VARCHAR,
|
||||||
password VARCHAR,
|
ecuuid UUID,
|
||||||
event_date TIMESTAMP WITHOUT TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
username VARCHAR,
|
||||||
ip VARCHAR,
|
password VARCHAR,
|
||||||
error TEXT,
|
event_date TIMESTAMP WITHOUT TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||||
event TEXT,
|
ip VARCHAR,
|
||||||
data_binary BYTEA,
|
error TEXT,
|
||||||
data TEXT,
|
event TEXT,
|
||||||
page VARCHAR,
|
data_binary BYTEA,
|
||||||
action VARCHAR,
|
data TEXT,
|
||||||
action_id VARCHAR,
|
page VARCHAR,
|
||||||
action_yes VARCHAR,
|
action VARCHAR,
|
||||||
action_flag VARCHAR,
|
action_id VARCHAR,
|
||||||
action_menu VARCHAR,
|
action_sub_id VARCHAR,
|
||||||
action_loaded VARCHAR,
|
action_yes VARCHAR,
|
||||||
action_value VARCHAR,
|
action_flag VARCHAR,
|
||||||
action_type VARCHAR,
|
action_menu VARCHAR,
|
||||||
action_error VARCHAR,
|
action_loaded VARCHAR,
|
||||||
user_agent VARCHAR,
|
action_value VARCHAR,
|
||||||
referer VARCHAR,
|
action_type VARCHAR,
|
||||||
script_name VARCHAR,
|
action_error VARCHAR,
|
||||||
query_string VARCHAR,
|
user_agent VARCHAR,
|
||||||
server_name VARCHAR,
|
referer VARCHAR,
|
||||||
http_host VARCHAR,
|
script_name VARCHAR,
|
||||||
http_accept VARCHAR,
|
query_string VARCHAR,
|
||||||
http_accept_charset VARCHAR,
|
server_name VARCHAR,
|
||||||
http_accept_encoding VARCHAR,
|
http_host VARCHAR,
|
||||||
session_id VARCHAR
|
http_accept VARCHAR,
|
||||||
|
http_accept_charset VARCHAR,
|
||||||
|
http_accept_encoding VARCHAR,
|
||||||
|
session_id VARCHAR
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_log.sql
|
-- END: table/edit_log.sql
|
||||||
-- START: table/edit_log_overflow.sql
|
-- START: table/edit_log_overflow.sql
|
||||||
@@ -707,15 +712,15 @@ ALTER TABLE edit_log_overflow ADD CONSTRAINT edit_log_overflow_euid_fkey FOREIGN
|
|||||||
|
|
||||||
-- DROP TABLE edit_access;
|
-- DROP TABLE edit_access;
|
||||||
CREATE TABLE edit_access (
|
CREATE TABLE edit_access (
|
||||||
edit_access_id SERIAL PRIMARY KEY,
|
edit_access_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0,
|
enabled SMALLINT NOT NULL DEFAULT 0,
|
||||||
protected SMALLINT DEFAULT 0,
|
protected SMALLINT DEFAULT 0,
|
||||||
deleted SMALLINT DEFAULT 0,
|
deleted SMALLINT DEFAULT 0,
|
||||||
uid VARCHAR,
|
uid VARCHAR,
|
||||||
name VARCHAR UNIQUE,
|
name VARCHAR UNIQUE,
|
||||||
description VARCHAR,
|
description VARCHAR,
|
||||||
color VARCHAR,
|
color VARCHAR,
|
||||||
additional_acl JSONB
|
additional_acl JSONB
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_access.sql
|
-- END: table/edit_access.sql
|
||||||
-- START: table/edit_access_user.sql
|
-- START: table/edit_access_user.sql
|
||||||
@@ -728,15 +733,15 @@ CREATE TABLE edit_access (
|
|||||||
|
|
||||||
-- DROP TABLE edit_access_user;
|
-- DROP TABLE edit_access_user;
|
||||||
CREATE TABLE edit_access_user (
|
CREATE TABLE edit_access_user (
|
||||||
edit_access_user_id SERIAL PRIMARY KEY,
|
edit_access_user_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
edit_access_id INT NOT NULL,
|
edit_access_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_access_id) REFERENCES edit_access (edit_access_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_access_id) REFERENCES edit_access (edit_access_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_user_id INT NOT NULL,
|
edit_user_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_user_id) REFERENCES edit_user (edit_user_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_user_id) REFERENCES edit_user (edit_user_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_access_right_id INT NOT NULL,
|
edit_access_right_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_access_right_id) REFERENCES edit_access_right (edit_access_right_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
edit_default SMALLINT DEFAULT 0,
|
edit_default SMALLINT DEFAULT 0,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0
|
enabled SMALLINT NOT NULL DEFAULT 0
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
-- END: table/edit_access_user.sql
|
-- END: table/edit_access_user.sql
|
||||||
-- START: table/edit_access_data.sql
|
-- START: table/edit_access_data.sql
|
||||||
@@ -749,12 +754,12 @@ CREATE TABLE edit_access_user (
|
|||||||
|
|
||||||
-- DROP TABLE edit_access_data;
|
-- DROP TABLE edit_access_data;
|
||||||
CREATE TABLE edit_access_data (
|
CREATE TABLE edit_access_data (
|
||||||
edit_access_data_id SERIAL PRIMARY KEY,
|
edit_access_data_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
edit_access_id INT NOT NULL,
|
edit_access_id INT NOT NULL,
|
||||||
FOREIGN KEY (edit_access_id) REFERENCES edit_access (edit_access_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
FOREIGN KEY (edit_access_id) REFERENCES edit_access (edit_access_id) MATCH FULL ON DELETE CASCADE ON UPDATE CASCADE,
|
||||||
enabled SMALLINT NOT NULL DEFAULT 0,
|
enabled SMALLINT NOT NULL DEFAULT 0,
|
||||||
name VARCHAR,
|
name VARCHAR,
|
||||||
value VARCHAR
|
value VARCHAR
|
||||||
) INHERITS (edit_generic) WITHOUT OIDS;
|
) INHERITS (edit_generic) WITHOUT OIDS;
|
||||||
|
|
||||||
-- create a unique index for each attached data block for each edit access can
|
-- create a unique index for each attached data block for each edit access can
|
||||||
|
|||||||
@@ -28,10 +28,10 @@ final class CoreLibsCheckFileTest extends TestCase
|
|||||||
public function filesList(): array
|
public function filesList(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
['filename.txt', 'txt', 5],
|
['filename.txt', 'txt', 5, 'text/plain'],
|
||||||
['filename.csv', 'csv', 15],
|
['filename.csv', 'csv', 15, 'text/csv'],
|
||||||
['filename.tsv', 'tsv', 0],
|
['filename.tsv', 'tsv', 0, 'text/plain'],
|
||||||
['file_does_not_exits', '', -1],
|
['file_does_not_exits', '', -1, ''],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,6 +63,15 @@ final class CoreLibsCheckFileTest extends TestCase
|
|||||||
return $list;
|
return $list;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function mimeTypeProvider(): array
|
||||||
|
{
|
||||||
|
$list = [];
|
||||||
|
foreach ($this->filesList() as $row) {
|
||||||
|
$list[$row[0] . ' must be mime type ' . $row[3]] = [$row[0], $row[3]];
|
||||||
|
}
|
||||||
|
return $list;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests if file extension matches
|
* Tests if file extension matches
|
||||||
*
|
*
|
||||||
@@ -115,6 +124,51 @@ final class CoreLibsCheckFileTest extends TestCase
|
|||||||
unlink($this->base_folder . $input);
|
unlink($this->base_folder . $input);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::getMimeType
|
||||||
|
* @dataProvider mimeTypeProvider
|
||||||
|
* @testdox getMimeType $input must be mime type $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $input
|
||||||
|
* @param string $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testGetMimeType(string $input, string $expected): void
|
||||||
|
{
|
||||||
|
if (!empty($expected)) {
|
||||||
|
$file = $this->base_folder . $input;
|
||||||
|
$fp = fopen($file, 'w');
|
||||||
|
switch ($expected) {
|
||||||
|
case 'text/csv':
|
||||||
|
for ($i = 1; $i <= 10; $i++) {
|
||||||
|
fwrite($fp, '"This is row","' . $expected . '",' . $i . PHP_EOL);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'text/tsv':
|
||||||
|
for ($i = 1; $i <= 10; $i++) {
|
||||||
|
fwrite($fp, "\"This is row\"\t\"" . $expected . "\"\t\"" . $i . PHP_EOL);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'text/plain':
|
||||||
|
fwrite($fp, 'This is mime type: ' . $expected . PHP_EOL);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
fclose($fp);
|
||||||
|
} else {
|
||||||
|
$this->expectException(\UnexpectedValueException::class);
|
||||||
|
}
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Check\File::getMimeType($this->base_folder . $input)
|
||||||
|
);
|
||||||
|
// unlink file
|
||||||
|
if (is_file($this->base_folder . $input)) {
|
||||||
|
unlink($this->base_folder . $input);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -1098,16 +1098,109 @@ final class CoreLibsCombinedArrayHandlerTest extends TestCase
|
|||||||
* @testdox arrayFlatForKey array $input will be $expected [$_dataName]
|
* @testdox arrayFlatForKey array $input will be $expected [$_dataName]
|
||||||
*
|
*
|
||||||
* @param array $input
|
* @param array $input
|
||||||
|
* @param string $search
|
||||||
* @param array $expected
|
* @param array $expected
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testArrayFlatForKey(array $input, $search, array $expected): void
|
public function testArrayFlatForKey(array $input, string $search, array $expected): void
|
||||||
{
|
{
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$expected,
|
$expected,
|
||||||
\CoreLibs\Combined\ArrayHandler::arrayFlatForKey($input, $search)
|
\CoreLibs\Combined\ArrayHandler::arrayFlatForKey($input, $search)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerArrayGetNextPrevKey(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'find, ok' => [
|
||||||
|
'input' => [
|
||||||
|
'a' => 'First',
|
||||||
|
'b' => 'Second',
|
||||||
|
'c' => 'Third',
|
||||||
|
],
|
||||||
|
'b',
|
||||||
|
'a',
|
||||||
|
'c'
|
||||||
|
],
|
||||||
|
'find, first' => [
|
||||||
|
'input' => [
|
||||||
|
'a' => 'First',
|
||||||
|
'b' => 'Second',
|
||||||
|
'c' => 'Third',
|
||||||
|
],
|
||||||
|
'a',
|
||||||
|
null,
|
||||||
|
'b'
|
||||||
|
],
|
||||||
|
'find, last' => [
|
||||||
|
'input' => [
|
||||||
|
'a' => 'First',
|
||||||
|
'b' => 'Second',
|
||||||
|
'c' => 'Third',
|
||||||
|
],
|
||||||
|
'c',
|
||||||
|
'b',
|
||||||
|
null
|
||||||
|
],
|
||||||
|
'find, not found' => [
|
||||||
|
'input' => [
|
||||||
|
'a' => 'First',
|
||||||
|
'b' => 'Second',
|
||||||
|
'c' => 'Third',
|
||||||
|
],
|
||||||
|
'z',
|
||||||
|
null,
|
||||||
|
null
|
||||||
|
],
|
||||||
|
'int, index' => [
|
||||||
|
'input' => [
|
||||||
|
'a',
|
||||||
|
'b',
|
||||||
|
'c'
|
||||||
|
],
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
2
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::arrayGetPrevKey, ::arrayGetNextKey
|
||||||
|
* @dataProvider providerArrayGetNextPrevKey
|
||||||
|
* @testdox arrayGetNextPrevKey get next/prev key for $search wtih $expected_prev/$expected_next [$_dataName]
|
||||||
|
*
|
||||||
|
* @param array $input
|
||||||
|
* @param int|string $search
|
||||||
|
* @param int|string|null $expected_prev
|
||||||
|
* @param int|string|null $expected_next
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testArrayGetNextPrevKey(
|
||||||
|
array $input,
|
||||||
|
int|string $search,
|
||||||
|
int|string|null $expected_prev,
|
||||||
|
int|string|null $expected_next
|
||||||
|
): void {
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected_prev,
|
||||||
|
\CoreLibs\Combined\ArrayHandler::arrayGetPrevKey($input, $search),
|
||||||
|
'Find prev key in array'
|
||||||
|
);
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected_next,
|
||||||
|
\CoreLibs\Combined\ArrayHandler::arrayGetNextKey($input, $search),
|
||||||
|
'Find next key in array'
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -66,6 +66,34 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* date string convert test
|
||||||
|
*
|
||||||
|
* @covers ::dateStringFormat
|
||||||
|
* @dataProvider timestampProvider
|
||||||
|
* @testdox dateStringFormat $input (microtime $flag) will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param int|float $input
|
||||||
|
* @param bool $flag
|
||||||
|
* @param string $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testDateStringFormat(
|
||||||
|
$input,
|
||||||
|
bool $flag_show_micro,
|
||||||
|
bool $flag_micro_as_float,
|
||||||
|
string $expected
|
||||||
|
): void {
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::dateStringFormat(
|
||||||
|
$input,
|
||||||
|
$flag_show_micro,
|
||||||
|
$flag_micro_as_float
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* interval for both directions
|
* interval for both directions
|
||||||
*
|
*
|
||||||
@@ -74,6 +102,11 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
public function intervalProvider(): array
|
public function intervalProvider(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
'on hour' => [
|
||||||
|
3600,
|
||||||
|
false,
|
||||||
|
'1h 0m 0s'
|
||||||
|
],
|
||||||
'interval no microtime' => [
|
'interval no microtime' => [
|
||||||
1641515890,
|
1641515890,
|
||||||
false,
|
false,
|
||||||
@@ -82,7 +115,7 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
'interval with microtime' => [
|
'interval with microtime' => [
|
||||||
1641515890,
|
1641515890,
|
||||||
true,
|
true,
|
||||||
'18999d 0h 38m 10s',
|
'18999d 0h 38m 10s 0ms',
|
||||||
],
|
],
|
||||||
'micro interval no microtime' => [
|
'micro interval no microtime' => [
|
||||||
1641515890.123456,
|
1641515890.123456,
|
||||||
@@ -92,7 +125,7 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
'micro interval with microtime' => [
|
'micro interval with microtime' => [
|
||||||
1641515890.123456,
|
1641515890.123456,
|
||||||
true,
|
true,
|
||||||
'18999d 0h 38m 10s 1235ms',
|
'18999d 0h 38m 10s 124ms',
|
||||||
],
|
],
|
||||||
'negative interval no microtime' => [
|
'negative interval no microtime' => [
|
||||||
-1641515890,
|
-1641515890,
|
||||||
@@ -103,27 +136,27 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
'microtime only' => [
|
'microtime only' => [
|
||||||
0.123456,
|
0.123456,
|
||||||
true,
|
true,
|
||||||
'0s 1235ms',
|
'0s 123ms',
|
||||||
],
|
],
|
||||||
'seconds only' => [
|
'seconds only' => [
|
||||||
30.123456,
|
30.123456,
|
||||||
true,
|
true,
|
||||||
'30s 1235ms',
|
'30s 123ms',
|
||||||
],
|
],
|
||||||
'minutes only' => [
|
'minutes only' => [
|
||||||
90.123456,
|
90.123456,
|
||||||
true,
|
true,
|
||||||
'1m 30s 1235ms',
|
'1m 30s 123ms',
|
||||||
],
|
],
|
||||||
'hours only' => [
|
'hours only' => [
|
||||||
3690.123456,
|
3690.123456,
|
||||||
true,
|
true,
|
||||||
'1h 1m 30s 1235ms',
|
'1h 1m 30s 123ms',
|
||||||
],
|
],
|
||||||
'days only' => [
|
'days only' => [
|
||||||
90090.123456,
|
90090.123456,
|
||||||
true,
|
true,
|
||||||
'1d 1h 1m 30s 1235ms',
|
'1d 1h 1m 30s 123ms',
|
||||||
],
|
],
|
||||||
'already set' => [
|
'already set' => [
|
||||||
'1d 1h 1m 30s 1235ms',
|
'1d 1h 1m 30s 1235ms',
|
||||||
@@ -143,6 +176,306 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* time seconds convert test
|
||||||
|
*
|
||||||
|
* @covers ::timeStringFormat
|
||||||
|
* @dataProvider intervalProvider
|
||||||
|
* @testdox timeStringFormat $input (microtime $flag) will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string|int|float $input
|
||||||
|
* @param bool $flag
|
||||||
|
* @param string $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testTimeStringFormat(string|int|float $input, bool $flag, string $expected): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::timeStringFormat($input, $flag)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* interval seconds convert
|
||||||
|
*
|
||||||
|
* @covers ::intervalStringFormat
|
||||||
|
* @dataProvider intervalProvider
|
||||||
|
* @testdox intervalStringFormat $input (microtime $show_micro) will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string|int|float $input
|
||||||
|
* @param bool $show_micro
|
||||||
|
* @param string $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testIntervalStringFormat(string|int|float $input, bool $show_micro, string $expected): void
|
||||||
|
{
|
||||||
|
// we skip string input, that is not allowed
|
||||||
|
if (is_string($input)) {
|
||||||
|
$this->assertTrue(true, 'Skip strings');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// invalid values throw exception in default
|
||||||
|
if ($input == 999999999999999) {
|
||||||
|
$this->expectException(\LengthException::class);
|
||||||
|
}
|
||||||
|
// below is equal to timeStringFormat
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::intervalStringFormat(
|
||||||
|
$input,
|
||||||
|
show_microseconds: $show_micro,
|
||||||
|
show_only_days: true,
|
||||||
|
skip_zero: false,
|
||||||
|
skip_last_zero: false,
|
||||||
|
truncate_nanoseconds: true,
|
||||||
|
truncate_zero_seconds_if_microseconds: false
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function intervalExtendedProvider(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
// A
|
||||||
|
'(60) default value' => [
|
||||||
|
[
|
||||||
|
'seconds' => 60,
|
||||||
|
],
|
||||||
|
'expected' => '1m',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
'(60) default value, skip_last_zero:false' => [
|
||||||
|
[
|
||||||
|
'seconds' => 60,
|
||||||
|
'skip_last_zero' => false,
|
||||||
|
],
|
||||||
|
'expected' => '1m 0s 0ms',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
// B
|
||||||
|
'(120.1) default value' => [
|
||||||
|
[
|
||||||
|
'seconds' => 120.1,
|
||||||
|
],
|
||||||
|
'expected' => '2m 100ms',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
'(120.1) default value, skip_zero:false' => [
|
||||||
|
[
|
||||||
|
'seconds' => 120.1,
|
||||||
|
'skip_zero' => false,
|
||||||
|
],
|
||||||
|
'expected' => '2m 0s 100ms',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
'(120.1) default value, skip_last_zero:false' => [
|
||||||
|
[
|
||||||
|
'seconds' => 120.1,
|
||||||
|
'skip_last_zero' => false,
|
||||||
|
],
|
||||||
|
'expected' => '2m 100ms',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
// C
|
||||||
|
'(3601) default value' => [
|
||||||
|
[
|
||||||
|
'seconds' => 3601,
|
||||||
|
],
|
||||||
|
'expected' => '1h 1s',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
'(3601) default value, skip_zero:false' => [
|
||||||
|
[
|
||||||
|
'seconds' => 3601,
|
||||||
|
'skip_zero' => false,
|
||||||
|
],
|
||||||
|
'expected' => '1h 0m 1s',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
'(3601) default value, skip_last_zero:false' => [
|
||||||
|
[
|
||||||
|
'seconds' => 3601,
|
||||||
|
'skip_last_zero' => false,
|
||||||
|
],
|
||||||
|
'expected' => '1h 1s 0ms',
|
||||||
|
'exception' => null
|
||||||
|
],
|
||||||
|
// TODO create unit tests for ALL edge cases
|
||||||
|
// CREATE abort tests, simple, all others are handled in exception tests
|
||||||
|
'exception: \UnexpectedValueException:1' => [
|
||||||
|
[
|
||||||
|
'seconds' => 99999999999999999999999
|
||||||
|
],
|
||||||
|
'expected' => null,
|
||||||
|
'exception' => [
|
||||||
|
'class' => \UnexpectedValueException::class,
|
||||||
|
'code' => 1,
|
||||||
|
],
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test all options for interval conversion
|
||||||
|
*
|
||||||
|
* @covers ::intervalStringFormat
|
||||||
|
* @dataProvider intervalExtendedProvider
|
||||||
|
* @testdox intervalStringFormat $input will be $expected / $exception [$_dataName]
|
||||||
|
*
|
||||||
|
* @param array<string,null|int|float|bool> $parameter_list
|
||||||
|
* @param string $expected
|
||||||
|
* @param array<string,mixed> $exception
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testExtendedIntervalStringFormat(
|
||||||
|
array $parameter_list,
|
||||||
|
?string $expected,
|
||||||
|
?array $exception
|
||||||
|
): void {
|
||||||
|
if ($expected === null && $exception === null) {
|
||||||
|
$this->assertFalse(true, 'Cannot have expected and exception null in test data');
|
||||||
|
}
|
||||||
|
$parameters = [];
|
||||||
|
foreach (
|
||||||
|
[
|
||||||
|
'seconds' => null,
|
||||||
|
'truncate_after' => '',
|
||||||
|
'natural_seperator' => false,
|
||||||
|
'name_space_seperator' => false,
|
||||||
|
'show_microseconds' => true,
|
||||||
|
'short_time_name' => true,
|
||||||
|
'skip_last_zero' => true,
|
||||||
|
'skip_zero' => true,
|
||||||
|
'show_only_days' => false,
|
||||||
|
'auto_fix_microseconds' => false,
|
||||||
|
'truncate_nanoseconds' => false,
|
||||||
|
'truncate_zero_seconds_if_microseconds' => true,
|
||||||
|
] as $param => $default
|
||||||
|
) {
|
||||||
|
if (empty($parameter_list[$param]) && $default === null) {
|
||||||
|
$this->assertFalse(true, 'Parameter ' . $param . ' is mandatory ');
|
||||||
|
} elseif (!isset($parameter_list[$param]) || $parameter_list[$param] === null) {
|
||||||
|
$parameters[] = $default;
|
||||||
|
} else {
|
||||||
|
$parameters[] = $parameter_list[$param];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($expected !== null) {
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
call_user_func_array('CoreLibs\Combined\DateTime::intervalStringFormat', $parameters)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
if (empty($exception['class']) || empty($exception['code'])) {
|
||||||
|
$this->assertFalse(true, 'Exception tests need Exception name and Code');
|
||||||
|
}
|
||||||
|
$this->expectException($exception['class']);
|
||||||
|
$this->expectExceptionCode($exception['code']);
|
||||||
|
// if we have a message, must be regex
|
||||||
|
if (!empty($exception['message'])) {
|
||||||
|
$this->expectExceptionMessageMatches($exception['message']);
|
||||||
|
}
|
||||||
|
call_user_func_array('CoreLibs\Combined\DateTime::intervalStringFormat', $parameters);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array<mixed>
|
||||||
|
*/
|
||||||
|
public function exceptionsIntervalProvider(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'UnexpectedValueException: 1 A' => [
|
||||||
|
'seconds' => 99999999999999999999999,
|
||||||
|
'params' => [],
|
||||||
|
'exception' => \UnexpectedValueException::class,
|
||||||
|
'exception_message' => "/^Seconds value is invalid, too large or more than six decimals: /",
|
||||||
|
'excpetion_code' => 1,
|
||||||
|
],
|
||||||
|
'UnexpectedValueException: 1 B' => [
|
||||||
|
'seconds' => 123.1234567,
|
||||||
|
'params' => [],
|
||||||
|
'exception' => \UnexpectedValueException::class,
|
||||||
|
'exception_message' => "/^Seconds value is invalid, too large or more than six decimals: /",
|
||||||
|
'excpetion_code' => 1,
|
||||||
|
],
|
||||||
|
// exception 2 is very likely covered by exception 1
|
||||||
|
'LengthException: 3' => [
|
||||||
|
'seconds' => 999999999999999999,
|
||||||
|
'params' => [
|
||||||
|
'show_only_days',
|
||||||
|
],
|
||||||
|
'exception' => \LengthException::class,
|
||||||
|
'exception_message' => "/^Input seconds value is too large for days output: /",
|
||||||
|
'excpetion_code' => 3,
|
||||||
|
],
|
||||||
|
'UnexpectedValueException: 4' => [
|
||||||
|
'seconds' => 1234567,
|
||||||
|
'params' => [
|
||||||
|
'truncate_after'
|
||||||
|
],
|
||||||
|
'exception' => \UnexpectedValueException::class,
|
||||||
|
'exception_message' => "/^truncate_after has an invalid value: /",
|
||||||
|
'excpetion_code' => 4,
|
||||||
|
],
|
||||||
|
'UnexpectedValueException: 5' => [
|
||||||
|
'seconds' => 1234567,
|
||||||
|
'params' => [
|
||||||
|
'show_only_days:truncate_after'
|
||||||
|
],
|
||||||
|
'exception' => \UnexpectedValueException::class,
|
||||||
|
'exception_message' =>
|
||||||
|
"/^If show_only_days is turned on, the truncate_after cannot be years or months: /",
|
||||||
|
'excpetion_code' => 5,
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test all exceptions
|
||||||
|
*
|
||||||
|
* @covers ::intervalStringFormat
|
||||||
|
* @dataProvider exceptionsIntervalProvider
|
||||||
|
* @testdox intervalStringFormat: test Exceptions
|
||||||
|
*
|
||||||
|
* @param int|float $seconds
|
||||||
|
* @param array<string> $params
|
||||||
|
* @param string $exception
|
||||||
|
* @param string $exception_message
|
||||||
|
* @param int $excpetion_code
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testExceptionsIntervalStringFormat(
|
||||||
|
int|float $seconds,
|
||||||
|
array $params,
|
||||||
|
string $exception,
|
||||||
|
string $exception_message,
|
||||||
|
int $excpetion_code,
|
||||||
|
): void {
|
||||||
|
$this->expectException($exception);
|
||||||
|
$this->expectExceptionMessageMatches($exception_message);
|
||||||
|
$this->expectExceptionCode($excpetion_code);
|
||||||
|
if (empty($params)) {
|
||||||
|
\CoreLibs\Combined\DateTime::intervalStringFormat($seconds);
|
||||||
|
} else {
|
||||||
|
if (in_array('show_only_days', $params)) {
|
||||||
|
\CoreLibs\Combined\DateTime::intervalStringFormat($seconds, show_only_days:true);
|
||||||
|
} elseif (in_array('truncate_after', $params)) {
|
||||||
|
\CoreLibs\Combined\DateTime::intervalStringFormat($seconds, truncate_after: 'v');
|
||||||
|
} elseif (in_array('show_only_days:truncate_after', $params)) {
|
||||||
|
\CoreLibs\Combined\DateTime::intervalStringFormat($seconds, show_only_days:true, truncate_after: 'y');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -203,6 +536,25 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::stringToTime
|
||||||
|
* @dataProvider reverseIntervalProvider
|
||||||
|
* @testdox stringToTime $input will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string|int|float $input
|
||||||
|
* @param string|int|float $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testStringToTime($input, $expected): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::stringToTime($input)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -238,6 +590,25 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::checkDate
|
||||||
|
* @dataProvider dateProvider
|
||||||
|
* @testdox checkDate $input will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $input
|
||||||
|
* @param bool $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testCheckDate(string $input, bool $expected): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::checkDate($input)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -297,6 +668,25 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::checkDateTime
|
||||||
|
* @dataProvider dateTimeProvider
|
||||||
|
* @testdox checkDateTime $input will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $input
|
||||||
|
* @param bool $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testCheckDateTime(string $input, bool $expected): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::checkDateTime($input)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -371,6 +761,37 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::compareDate
|
||||||
|
* @dataProvider dateCompareProvider
|
||||||
|
* @testdox compareDate $input_a compared to $input_b will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $input_a
|
||||||
|
* @param string $input_b
|
||||||
|
* @param int|bool $expected
|
||||||
|
* @param string|null $exception
|
||||||
|
* @param int|null $exception_code
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testCompareDate(
|
||||||
|
string $input_a,
|
||||||
|
string $input_b,
|
||||||
|
int|bool $expected,
|
||||||
|
?string $exception,
|
||||||
|
?int $exception_code
|
||||||
|
): void {
|
||||||
|
if ($expected === false) {
|
||||||
|
$this->expectException($exception);
|
||||||
|
$this->expectExceptionCode($exception_code);
|
||||||
|
}
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::compareDate($input_a, $input_b)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -466,6 +887,37 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::compareDateTime
|
||||||
|
* @dataProvider dateTimeCompareProvider
|
||||||
|
* @testdox compareDateTime $input_a compared to $input_b will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $input_a
|
||||||
|
* @param string $input_b
|
||||||
|
* @param int|bool $expected
|
||||||
|
* @param string|null $exception
|
||||||
|
* @param int|null $exception_code
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testCompareDateTime(
|
||||||
|
string $input_a,
|
||||||
|
string $input_b,
|
||||||
|
int|bool $expected,
|
||||||
|
?string $exception,
|
||||||
|
?int $exception_code
|
||||||
|
): void {
|
||||||
|
if ($expected === false) {
|
||||||
|
$this->expectException($exception);
|
||||||
|
$this->expectExceptionCode($exception_code);
|
||||||
|
}
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Combined\DateTime::compareDateTime($input_a, $input_b)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -520,214 +972,6 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public function dateRangeHasWeekendProvider(): array
|
|
||||||
{
|
|
||||||
return [
|
|
||||||
'no weekend' => [
|
|
||||||
'2023-07-03',
|
|
||||||
'2023-07-04',
|
|
||||||
false
|
|
||||||
],
|
|
||||||
'start weekend sat' => [
|
|
||||||
'2023-07-01',
|
|
||||||
'2023-07-04',
|
|
||||||
true
|
|
||||||
],
|
|
||||||
'start weekend sun' => [
|
|
||||||
'2023-07-02',
|
|
||||||
'2023-07-04',
|
|
||||||
true
|
|
||||||
],
|
|
||||||
'end weekend sat' => [
|
|
||||||
'2023-07-03',
|
|
||||||
'2023-07-08',
|
|
||||||
true
|
|
||||||
],
|
|
||||||
'end weekend sun' => [
|
|
||||||
'2023-07-03',
|
|
||||||
'2023-07-09',
|
|
||||||
true
|
|
||||||
],
|
|
||||||
'long period > 6 days' => [
|
|
||||||
'2023-07-03',
|
|
||||||
'2023-07-27',
|
|
||||||
true
|
|
||||||
]
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* date string convert test
|
|
||||||
*
|
|
||||||
* @covers ::dateStringFormat
|
|
||||||
* @dataProvider timestampProvider
|
|
||||||
* @testdox dateStringFormat $input (microtime $flag) will be $expected [$_dataName]
|
|
||||||
*
|
|
||||||
* @param int|float $input
|
|
||||||
* @param bool $flag
|
|
||||||
* @param string $expected
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function testDateStringFormat(
|
|
||||||
$input,
|
|
||||||
bool $flag_show_micro,
|
|
||||||
bool $flag_micro_as_float,
|
|
||||||
string $expected
|
|
||||||
): void {
|
|
||||||
$this->assertEquals(
|
|
||||||
$expected,
|
|
||||||
\CoreLibs\Combined\DateTime::dateStringFormat(
|
|
||||||
$input,
|
|
||||||
$flag_show_micro,
|
|
||||||
$flag_micro_as_float
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* interval convert test
|
|
||||||
*
|
|
||||||
* @covers ::timeStringFormat
|
|
||||||
* @dataProvider intervalProvider
|
|
||||||
* @testdox timeStringFormat $input (microtime $flag) will be $expected [$_dataName]
|
|
||||||
*
|
|
||||||
* @param int|float $input
|
|
||||||
* @param bool $flag
|
|
||||||
* @param string $expected
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function testTimeStringFormat($input, bool $flag, string $expected): void
|
|
||||||
{
|
|
||||||
$this->assertEquals(
|
|
||||||
$expected,
|
|
||||||
\CoreLibs\Combined\DateTime::timeStringFormat($input, $flag)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @covers ::stringToTime
|
|
||||||
* @dataProvider reverseIntervalProvider
|
|
||||||
* @testdox stringToTime $input will be $expected [$_dataName]
|
|
||||||
*
|
|
||||||
* @param string|int|float $input
|
|
||||||
* @param string|int|float $expected
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function testStringToTime($input, $expected): void
|
|
||||||
{
|
|
||||||
$this->assertEquals(
|
|
||||||
$expected,
|
|
||||||
\CoreLibs\Combined\DateTime::stringToTime($input)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @covers ::checkDate
|
|
||||||
* @dataProvider dateProvider
|
|
||||||
* @testdox checkDate $input will be $expected [$_dataName]
|
|
||||||
*
|
|
||||||
* @param string $input
|
|
||||||
* @param bool $expected
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function testCheckDate(string $input, bool $expected): void
|
|
||||||
{
|
|
||||||
$this->assertEquals(
|
|
||||||
$expected,
|
|
||||||
\CoreLibs\Combined\DateTime::checkDate($input)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @covers ::checkDateTime
|
|
||||||
* @dataProvider dateTimeProvider
|
|
||||||
* @testdox checkDateTime $input will be $expected [$_dataName]
|
|
||||||
*
|
|
||||||
* @param string $input
|
|
||||||
* @param bool $expected
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function testCheckDateTime(string $input, bool $expected): void
|
|
||||||
{
|
|
||||||
$this->assertEquals(
|
|
||||||
$expected,
|
|
||||||
\CoreLibs\Combined\DateTime::checkDateTime($input)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @covers ::compareDate
|
|
||||||
* @dataProvider dateCompareProvider
|
|
||||||
* @testdox compareDate $input_a compared to $input_b will be $expected [$_dataName]
|
|
||||||
*
|
|
||||||
* @param string $input_a
|
|
||||||
* @param string $input_b
|
|
||||||
* @param int|bool $expected
|
|
||||||
* @param string|null $exception
|
|
||||||
* @param int|null $exception_code
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function testCompareDate(
|
|
||||||
string $input_a,
|
|
||||||
string $input_b,
|
|
||||||
int|bool $expected,
|
|
||||||
?string $exception,
|
|
||||||
?int $exception_code
|
|
||||||
): void {
|
|
||||||
if ($expected === false) {
|
|
||||||
$this->expectException($exception);
|
|
||||||
$this->expectExceptionCode($exception_code);
|
|
||||||
}
|
|
||||||
$this->assertEquals(
|
|
||||||
$expected,
|
|
||||||
\CoreLibs\Combined\DateTime::compareDate($input_a, $input_b)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Undocumented function
|
|
||||||
*
|
|
||||||
* @covers ::compareDateTime
|
|
||||||
* @dataProvider dateTimeCompareProvider
|
|
||||||
* @testdox compareDateTime $input_a compared to $input_b will be $expected [$_dataName]
|
|
||||||
*
|
|
||||||
* @param string $input_a
|
|
||||||
* @param string $input_b
|
|
||||||
* @param int|bool $expected
|
|
||||||
* @param string|null $exception
|
|
||||||
* @param int|null $exception_code
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function testCompareDateTime(
|
|
||||||
string $input_a,
|
|
||||||
string $input_b,
|
|
||||||
int|bool $expected,
|
|
||||||
?string $exception,
|
|
||||||
?int $exception_code
|
|
||||||
): void {
|
|
||||||
if ($expected === false) {
|
|
||||||
$this->expectException($exception);
|
|
||||||
$this->expectExceptionCode($exception_code);
|
|
||||||
}
|
|
||||||
$this->assertEquals(
|
|
||||||
$expected,
|
|
||||||
\CoreLibs\Combined\DateTime::compareDateTime($input_a, $input_b)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -906,6 +1150,47 @@ final class CoreLibsCombinedDateTimeTest extends TestCase
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function dateRangeHasWeekendProvider(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'no weekend' => [
|
||||||
|
'2023-07-03',
|
||||||
|
'2023-07-04',
|
||||||
|
false
|
||||||
|
],
|
||||||
|
'start weekend sat' => [
|
||||||
|
'2023-07-01',
|
||||||
|
'2023-07-04',
|
||||||
|
true
|
||||||
|
],
|
||||||
|
'start weekend sun' => [
|
||||||
|
'2023-07-02',
|
||||||
|
'2023-07-04',
|
||||||
|
true
|
||||||
|
],
|
||||||
|
'end weekend sat' => [
|
||||||
|
'2023-07-03',
|
||||||
|
'2023-07-08',
|
||||||
|
true
|
||||||
|
],
|
||||||
|
'end weekend sun' => [
|
||||||
|
'2023-07-03',
|
||||||
|
'2023-07-09',
|
||||||
|
true
|
||||||
|
],
|
||||||
|
'long period > 6 days' => [
|
||||||
|
'2023-07-03',
|
||||||
|
'2023-07-27',
|
||||||
|
true
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -253,7 +253,8 @@ final class CoreLibsConvertByteTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testHumanReadableByteFormatException(int $flag): void
|
public function testHumanReadableByteFormatException(int $flag): void
|
||||||
{
|
{
|
||||||
$this->expectException(\Exception::class);
|
$this->expectException(\InvalidArgumentException::class);
|
||||||
|
$this->expectExceptionCode(1);
|
||||||
\CoreLibs\Convert\Byte::humanReadableByteFormat(12, $flag);
|
\CoreLibs\Convert\Byte::humanReadableByteFormat(12, $flag);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -272,7 +273,8 @@ final class CoreLibsConvertByteTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testStringByteFormatException(int $flag): void
|
public function testStringByteFormatException(int $flag): void
|
||||||
{
|
{
|
||||||
$this->expectException(\Exception::class);
|
$this->expectException(\InvalidArgumentException::class);
|
||||||
|
$this->expectExceptionCode(1);
|
||||||
\CoreLibs\Convert\Byte::stringByteFormat(12, $flag);
|
\CoreLibs\Convert\Byte::stringByteFormat(12, $flag);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
1186
test/phpunit/Convert/CoreLibsConvertColorTest.php
Normal file
1186
test/phpunit/Convert/CoreLibsConvertColorTest.php
Normal file
File diff suppressed because it is too large
Load Diff
@@ -9,7 +9,7 @@ use PHPUnit\Framework\TestCase;
|
|||||||
/**
|
/**
|
||||||
* Test class for Convert\Colors
|
* Test class for Convert\Colors
|
||||||
* @coversDefaultClass \CoreLibs\Convert\Colors
|
* @coversDefaultClass \CoreLibs\Convert\Colors
|
||||||
* @testdox \CoreLibs\Convert\Colors method tests
|
* @testdox \CoreLibs\Convert\Colors legacy method tests
|
||||||
*/
|
*/
|
||||||
final class CoreLibsConvertColorsTest extends TestCase
|
final class CoreLibsConvertColorsTest extends TestCase
|
||||||
{
|
{
|
||||||
@@ -21,7 +21,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function rgb2hexColorProvider(): array
|
public function providerRgb2hexColor(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'color' => [
|
'color' => [
|
||||||
@@ -88,7 +88,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function hex2rgbColorProvider(): array
|
public function providerHex2rgbColor(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'color' => [
|
'color' => [
|
||||||
@@ -215,7 +215,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function rgb2hsbColorProvider(): array
|
public function providerRgb2hsbColor(): array
|
||||||
{
|
{
|
||||||
$list = [];
|
$list = [];
|
||||||
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
||||||
@@ -234,7 +234,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function hsb2rgbColorProvider(): array
|
public function providerHsb2rgbColor(): array
|
||||||
{
|
{
|
||||||
$list = [];
|
$list = [];
|
||||||
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
||||||
@@ -253,7 +253,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function rgb2hslColorProvider(): array
|
public function providerRgb2hslColor(): array
|
||||||
{
|
{
|
||||||
$list = [];
|
$list = [];
|
||||||
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
||||||
@@ -272,7 +272,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function hsl2rgbColorProvider(): array
|
public function providerHsl2rgbColor(): array
|
||||||
{
|
{
|
||||||
$list = [];
|
$list = [];
|
||||||
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
foreach ($this->rgb2hslAndhsbList() as $name => $values) {
|
||||||
@@ -291,7 +291,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
* TODO: add cross convert check
|
* TODO: add cross convert check
|
||||||
*
|
*
|
||||||
* @covers ::rgb2hex
|
* @covers ::rgb2hex
|
||||||
* @dataProvider rgb2hexColorProvider
|
* @dataProvider providerRgb2hexColor
|
||||||
* @testdox rgb2hex $input_r,$input_g,$input_b will be $expected [$_dataName]
|
* @testdox rgb2hex $input_r,$input_g,$input_b will be $expected [$_dataName]
|
||||||
*
|
*
|
||||||
* @param int $input_r
|
* @param int $input_r
|
||||||
@@ -342,7 +342,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::hex2rgb
|
* @covers ::hex2rgb
|
||||||
* @dataProvider hex2rgbColorProvider
|
* @dataProvider providerHex2rgbColor
|
||||||
* @testdox hex2rgb $input will be $expected, $expected_str str[,], $expected_str_sep str[$separator] [$_dataName]
|
* @testdox hex2rgb $input will be $expected, $expected_str str[,], $expected_str_sep str[$separator] [$_dataName]
|
||||||
*
|
*
|
||||||
* @param string $input
|
* @param string $input
|
||||||
@@ -385,7 +385,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::rgb2hsb
|
* @covers ::rgb2hsb
|
||||||
* @dataProvider rgb2hsbColorProvider
|
* @dataProvider providerRgb2hsbColor
|
||||||
* @testdox rgb2hsb $input_r,$input_g,$input_b will be $expected [$_dataName]
|
* @testdox rgb2hsb $input_r,$input_g,$input_b will be $expected [$_dataName]
|
||||||
*
|
*
|
||||||
* @param integer $input_r
|
* @param integer $input_r
|
||||||
@@ -409,7 +409,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::hsb2rgb
|
* @covers ::hsb2rgb
|
||||||
* @dataProvider hsb2rgbColorProvider
|
* @dataProvider providerHsb2rgbColor
|
||||||
* @testdox hsb2rgb $input_h,$input_s,$input_b will be $expected [$_dataName]
|
* @testdox hsb2rgb $input_h,$input_s,$input_b will be $expected [$_dataName]
|
||||||
*
|
*
|
||||||
* @param float $input_h
|
* @param float $input_h
|
||||||
@@ -434,7 +434,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::rgb2hsl
|
* @covers ::rgb2hsl
|
||||||
* @dataProvider rgb2hslColorProvider
|
* @dataProvider providerRgb2hslColor
|
||||||
* @testdox rgb2hsl $input_r,$input_g,$input_b will be $expected [$_dataName]
|
* @testdox rgb2hsl $input_r,$input_g,$input_b will be $expected [$_dataName]
|
||||||
*
|
*
|
||||||
* @param integer $input_r
|
* @param integer $input_r
|
||||||
@@ -458,7 +458,7 @@ final class CoreLibsConvertColorsTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::hsl2rgb
|
* @covers ::hsl2rgb
|
||||||
* @dataProvider hsl2rgbColorProvider
|
* @dataProvider providerHsl2rgbColor
|
||||||
* @testdox hsl2rgb $input_h,$input_s,$input_l will be $expected [$_dataName]
|
* @testdox hsl2rgb $input_h,$input_s,$input_l will be $expected [$_dataName]
|
||||||
*
|
*
|
||||||
* @param integer|float $input_h
|
* @param integer|float $input_h
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ final class CoreLibsConvertMathTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array<mixed>
|
* @return array<mixed>
|
||||||
*/
|
*/
|
||||||
public function fceilProvider(): array
|
public function providerFceil(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'5.5 must be 6' => [5.5, 6],
|
'5.5 must be 6' => [5.5, 6],
|
||||||
@@ -31,7 +31,7 @@ final class CoreLibsConvertMathTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::fceil
|
* @covers ::fceil
|
||||||
* @dataProvider fceilProvider
|
* @dataProvider providerFceil
|
||||||
* @testdox fceil: Input $input must be $expected
|
* @testdox fceil: Input $input must be $expected
|
||||||
*
|
*
|
||||||
* @param float $input
|
* @param float $input
|
||||||
@@ -51,7 +51,7 @@ final class CoreLibsConvertMathTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array<mixed>
|
* @return array<mixed>
|
||||||
*/
|
*/
|
||||||
public function floorProvider(): array
|
public function providerFloor(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'5123456 with -3 must be 5123000' => [5123456, -3, 5123000],
|
'5123456 with -3 must be 5123000' => [5123456, -3, 5123000],
|
||||||
@@ -63,7 +63,7 @@ final class CoreLibsConvertMathTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::floorp
|
* @covers ::floorp
|
||||||
* @dataProvider floorProvider
|
* @dataProvider providerFloor
|
||||||
* @testdox floor: Input $input with cutoff $cutoff must be $expected
|
* @testdox floor: Input $input with cutoff $cutoff must be $expected
|
||||||
*
|
*
|
||||||
* @param int $input
|
* @param int $input
|
||||||
@@ -84,7 +84,7 @@ final class CoreLibsConvertMathTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array<mixed>
|
* @return array<mixed>
|
||||||
*/
|
*/
|
||||||
public function initNumericProvider(): array
|
public function providerInitNumeric(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'5 must be 5' => [5, 5, 'int'],
|
'5 must be 5' => [5, 5, 'int'],
|
||||||
@@ -98,7 +98,7 @@ final class CoreLibsConvertMathTest extends TestCase
|
|||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::initNumeric
|
* @covers ::initNumeric
|
||||||
* @dataProvider initNumericProvider
|
* @dataProvider providerInitNumeric
|
||||||
* @testdox initNumeric: Input $info $input must match $expected [$_dataName]
|
* @testdox initNumeric: Input $info $input must match $expected [$_dataName]
|
||||||
*
|
*
|
||||||
* @param int|float|string $input
|
* @param int|float|string $input
|
||||||
@@ -113,6 +113,388 @@ final class CoreLibsConvertMathTest extends TestCase
|
|||||||
\CoreLibs\Convert\Math::initNumeric($input)
|
\CoreLibs\Convert\Math::initNumeric($input)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerCbrt(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'cube root of 2' => [2, 1.25992, 5],
|
||||||
|
'cube root of 3' => [3, 1.44225, 5],
|
||||||
|
'cube root of -1' => [-1, 'NAN', 0],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::cbrt
|
||||||
|
* @dataProvider providerCbrt
|
||||||
|
* @testdox initNumeric: Input $input must match $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param float|int $number
|
||||||
|
* @param float $expected
|
||||||
|
* @param int $round_to
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testCbrt(float|int $number, float|string $expected, int $round_to): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
round(\CoreLibs\Convert\Math::cbrt($number), $round_to)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerMultiplyMatrices(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'[3] x [3] => [3x1]' => [
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
[14]
|
||||||
|
],
|
||||||
|
'[3] x [3x1]' => [
|
||||||
|
[1, 2, 3],
|
||||||
|
[[1], [2], [3]],
|
||||||
|
[14]
|
||||||
|
],
|
||||||
|
'[3] x [3x1]' => [
|
||||||
|
[1, 2, 3],
|
||||||
|
[[1], [2], [3]],
|
||||||
|
[14]
|
||||||
|
],
|
||||||
|
'[1x3L] x [3x1]' => [
|
||||||
|
[[1, 2, 3]],
|
||||||
|
[[1], [2], [3]],
|
||||||
|
[14]
|
||||||
|
],
|
||||||
|
'[1x3] x [3x1]' => [
|
||||||
|
[[1], [2], [3]],
|
||||||
|
[[1], [2], [3]],
|
||||||
|
[1, 2, 3]
|
||||||
|
],
|
||||||
|
'[2x3] x [3] => [3x1]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3]
|
||||||
|
],
|
||||||
|
[1, 2, 3],
|
||||||
|
[
|
||||||
|
14,
|
||||||
|
14
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[2x3] x [3x1]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3]
|
||||||
|
],
|
||||||
|
[[1], [2], [3]],
|
||||||
|
[
|
||||||
|
14,
|
||||||
|
14
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[2x3] x [2x3] => [3x3]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[3, 6, 9],
|
||||||
|
[3, 6, 9]
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[2x3] x [3x3]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
[0, 0, 0],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[3, 6, 9],
|
||||||
|
[3, 6, 9]
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[2x3] x [3x2]' => [
|
||||||
|
'a' => [
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
'b' => [
|
||||||
|
[1, 1],
|
||||||
|
[2, 2],
|
||||||
|
[3, 3],
|
||||||
|
],
|
||||||
|
'prod' => [
|
||||||
|
[14, 14],
|
||||||
|
[14, 14],
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[3x3] x [3] => [1x3]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[1, 2, 3],
|
||||||
|
[
|
||||||
|
14,
|
||||||
|
14,
|
||||||
|
14
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[3x3] x [2x3] => [3x3]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[3, 6, 9],
|
||||||
|
[3, 6, 9],
|
||||||
|
[3, 6, 9],
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[3x3] x [3x3]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
// [0, 0, 0],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[3, 6, 9],
|
||||||
|
[3, 6, 9],
|
||||||
|
[3, 6, 9],
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[3] x [3x3]' => [
|
||||||
|
[1, 2, 3],
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[6, 12, 18],
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'[2x3] x [3x3]' => [
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
[1, 2, 3],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[6, 12, 18],
|
||||||
|
[6, 12, 18],
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'inblanaced [2x2,3] x [3x2]' => [
|
||||||
|
'a' => [
|
||||||
|
[1, 2, 3],
|
||||||
|
[4, 5]
|
||||||
|
],
|
||||||
|
'b' => [
|
||||||
|
[6, 7],
|
||||||
|
[8, 9],
|
||||||
|
[10, 11]
|
||||||
|
],
|
||||||
|
'result' => [
|
||||||
|
[52, 58],
|
||||||
|
[64, 73],
|
||||||
|
]
|
||||||
|
],
|
||||||
|
'inblanaced [2x3] x [3x1,2]' => [
|
||||||
|
'a' => [
|
||||||
|
[1, 2, 3],
|
||||||
|
[4, 5, 7]
|
||||||
|
],
|
||||||
|
'b' => [
|
||||||
|
[7, 8],
|
||||||
|
[9, 10],
|
||||||
|
[11]
|
||||||
|
],
|
||||||
|
'result' => [
|
||||||
|
[58, 28],
|
||||||
|
[150, 82],
|
||||||
|
]
|
||||||
|
],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::multiplyMatrices
|
||||||
|
* @dataProvider providerMultiplyMatrices
|
||||||
|
* @testdox initNumeric: Input $input_a x $input_b must match $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param array $input_a
|
||||||
|
* @param array $input_b
|
||||||
|
* @param array $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testMultiplyMatrices(array $input_a, array $input_b, array $expected): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Convert\Math::multiplyMatrices($input_a, $input_b)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerEqualWithEpsilon(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'equal' => [
|
||||||
|
'a' => 0.000000000000000222,
|
||||||
|
'b' => 0.000000000000000222,
|
||||||
|
'epsilon' => PHP_FLOAT_EPSILON,
|
||||||
|
'equal' => true,
|
||||||
|
],
|
||||||
|
'almost equal' => [
|
||||||
|
'a' => 0.000000000000000222,
|
||||||
|
'b' => 0.000000000000000232,
|
||||||
|
'epsilon' => PHP_FLOAT_EPSILON,
|
||||||
|
'equal' => true,
|
||||||
|
],
|
||||||
|
'not equal' => [
|
||||||
|
'a' => 0.000000000000000222,
|
||||||
|
'b' => 0.000000000000004222,
|
||||||
|
'epsilon' => PHP_FLOAT_EPSILON,
|
||||||
|
'equal' => false,
|
||||||
|
],
|
||||||
|
'equal, different epsilon' => [
|
||||||
|
'a' => 0.000000000000000222,
|
||||||
|
'b' => 0.000000000000004222,
|
||||||
|
'epsilon' => 0.0001,
|
||||||
|
'equal' => true,
|
||||||
|
],
|
||||||
|
'not equal, different epsilon' => [
|
||||||
|
'a' => 0.0001,
|
||||||
|
'b' => 0.0002,
|
||||||
|
'epsilon' => 0.0001,
|
||||||
|
'equal' => false,
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::equalWithEpsilon
|
||||||
|
* @dataProvider providerEqualWithEpsilon
|
||||||
|
* @testdox equalWithEpsilon with $a and $b and Epsilon: $epsilon must be equal: $equal [$_dataName]
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testEqualWithEpsilon(float $a, float $b, float $epsilon, bool $equal): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$equal,
|
||||||
|
\CoreLibs\Convert\Math::equalWithEpsilon($a, $b, $epsilon)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerCompareWithEpsilon(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'smaller, true' => [
|
||||||
|
'value' => 0.0001,
|
||||||
|
'compare' => '<',
|
||||||
|
'limit' => 0.0002,
|
||||||
|
'epsilon' => 0.00001,
|
||||||
|
'match' => true,
|
||||||
|
],
|
||||||
|
'smaller, false' => [
|
||||||
|
'value' => 0.0001,
|
||||||
|
'compare' => '<',
|
||||||
|
'limit' => 0.0001,
|
||||||
|
'epsilon' => 0.00001,
|
||||||
|
'match' => false,
|
||||||
|
],
|
||||||
|
'bigger, true' => [
|
||||||
|
'value' => 0.0002,
|
||||||
|
'compare' => '>',
|
||||||
|
'limit' => 0.0001,
|
||||||
|
'epsilon' => 0.00001,
|
||||||
|
'match' => true,
|
||||||
|
],
|
||||||
|
'bigger, false' => [
|
||||||
|
'value' => 0.0001,
|
||||||
|
'compare' => '>',
|
||||||
|
'limit' => 0.0001,
|
||||||
|
'epsilon' => 0.00001,
|
||||||
|
'match' => false,
|
||||||
|
],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::compareWithEpsilon
|
||||||
|
* @dataProvider providerCompareWithEpsilon
|
||||||
|
* @testdox compareWithEpsilon $value $compare $limit with $epsilon must match: $match [$_dataName]
|
||||||
|
*
|
||||||
|
* @param float $value
|
||||||
|
* @param string $compare
|
||||||
|
* @param float $limit
|
||||||
|
* @param float $epslion
|
||||||
|
* @param bool $match
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testCompareWithEpsilon(
|
||||||
|
float $value,
|
||||||
|
string $compare,
|
||||||
|
float $limit,
|
||||||
|
float $epsilon,
|
||||||
|
bool $match
|
||||||
|
): void {
|
||||||
|
$this->assertEquals(
|
||||||
|
$match,
|
||||||
|
\CoreLibs\Convert\Math::compareWithEpsilon($value, $compare, $limit, $epsilon)
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
@@ -33,15 +33,14 @@ final class CoreLibsConvertMimeEncodeTest extends TestCase
|
|||||||
'The quick brown fox jumps over the lazy sheep that sleeps in the ravine '
|
'The quick brown fox jumps over the lazy sheep that sleeps in the ravine '
|
||||||
. 'and has no idea what is going on here',
|
. 'and has no idea what is going on here',
|
||||||
'UTF-8',
|
'UTF-8',
|
||||||
'The quick brown fox jumps over the lazy sheep that sleeps in the ravine '
|
"The quick brown fox jumps over the lazy sheep that sleeps in the ravine and\r\n"
|
||||||
. 'and has no idea what is going on here'
|
. ' has no idea what is going on here'
|
||||||
],
|
],
|
||||||
'standard with special chars UTF-8' => [
|
'standard with special chars UTF-8' => [
|
||||||
'This is ümläßtと漢字もカタカナ!!^$%&',
|
'This is ümläßtと漢字もカタカナ!!^$%&',
|
||||||
'UTF-8',
|
'UTF-8',
|
||||||
'This is =?UTF-8?B?w7xtbMOkw59044Go5ryi5a2X44KC44Kr44K/44Kr44OK77yBIV4k?='
|
"This is =?UTF-8?B?w7xtbMOkw59044Go5ryi5a2X44KC44Kr44K/44Kr44OK77yBIV4k?=\r\n"
|
||||||
. "\r\n"
|
. ' =?UTF-8?B?JSY=?='
|
||||||
. ' =?UTF-8?B?JQ==?=&'
|
|
||||||
],
|
],
|
||||||
'35 chars and space at the end UTF-8' => [
|
'35 chars and space at the end UTF-8' => [
|
||||||
'12345678901234567890123456789012345 '
|
'12345678901234567890123456789012345 '
|
||||||
@@ -62,9 +61,8 @@ final class CoreLibsConvertMimeEncodeTest extends TestCase
|
|||||||
. 'is there a space?',
|
. 'is there a space?',
|
||||||
'UTF-8',
|
'UTF-8',
|
||||||
"=?UTF-8?B?44Kr44K/44Kr44OK44Kr44K/44Kr44OK44GL44Gq44Kr44K/44Kr44OK44Kr?=\r\n"
|
"=?UTF-8?B?44Kr44K/44Kr44OK44Kr44K/44Kr44OK44GL44Gq44Kr44K/44Kr44OK44Kr?=\r\n"
|
||||||
. " =?UTF-8?B?44K/44Kr44OK?=\r\n"
|
. " =?UTF-8?B?44K/44Kr44OK44GL44Gq44Kr44K/44Kr44OK44Kr44K/44Kr44OK44GL44Gq?=\r\n"
|
||||||
. " =?UTF-8?B?44GL44Gq44Kr44K/44Kr44OK44Kr44K/44Kr44OK44GL44Gq44Kr44K/44Kr?=\r\n"
|
. " =?UTF-8?B?44Kr44K/44Kr44OK44Kr44K/IGlzIHRoZXJlIGEgc3BhY2U/?="
|
||||||
. " =?UTF-8?B?44OK44Kr44K/?= is there a =?UTF-8?B?c3BhY2U/?="
|
|
||||||
]
|
]
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -85,16 +83,28 @@ final class CoreLibsConvertMimeEncodeTest extends TestCase
|
|||||||
// print "MIME: -" . $encoded . "-\n";
|
// print "MIME: -" . $encoded . "-\n";
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$expected,
|
$expected,
|
||||||
$encoded
|
$encoded,
|
||||||
|
"__mbMimeEncode"
|
||||||
);
|
);
|
||||||
$decoded = mb_decode_mimeheader($encoded);
|
$decoded = mb_decode_mimeheader($encoded);
|
||||||
// print "INPUT : " . $input . "\n";
|
// print "ENCODED: " . $encoded . "\n";
|
||||||
// print "DECODED: " . $decoded . "\n";
|
// print "INPUT : " . $input . " | " . mb_strlen($input) . "\n";
|
||||||
|
// print "DECODED: " . $decoded . " | " . mb_strlen($decoded) . "\n";
|
||||||
|
// $test_enc = mb_encode_mimeheader($input, $encoding);
|
||||||
|
// $test_dec = mb_decode_mimeheader($test_enc);
|
||||||
|
// print "TEST ENC: " . $test_enc . "\n";
|
||||||
// back compare decoded
|
// back compare decoded
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$input,
|
$input,
|
||||||
$decoded
|
$decoded,
|
||||||
|
"mb_decode_mimeheader"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// $this->assertEquals(
|
||||||
|
// $input,
|
||||||
|
// $test_dec,
|
||||||
|
// 'mb_encode_to_decode'
|
||||||
|
// );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -13,6 +13,8 @@ use PHPUnit\Framework\TestCase;
|
|||||||
*/
|
*/
|
||||||
final class CoreLibsConvertStringsTest extends TestCase
|
final class CoreLibsConvertStringsTest extends TestCase
|
||||||
{
|
{
|
||||||
|
private const DATA_FOLDER = __DIR__ . DIRECTORY_SEPARATOR . 'data' . DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Undocumented function
|
* Undocumented function
|
||||||
*
|
*
|
||||||
@@ -256,6 +258,126 @@ final class CoreLibsConvertStringsTest extends TestCase
|
|||||||
$output
|
$output
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* provider for testStripMultiplePathSlashes
|
||||||
|
*
|
||||||
|
* @return array<mixed>
|
||||||
|
*/
|
||||||
|
public function stripMultiplePathSlashesProvider(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'no slahses' => [
|
||||||
|
'input' => 'string_abc',
|
||||||
|
'expected' => 'string_abc',
|
||||||
|
],
|
||||||
|
'one slash' => [
|
||||||
|
'input' => 'some/foo',
|
||||||
|
'expected' => 'some/foo',
|
||||||
|
],
|
||||||
|
'two slashes' => [
|
||||||
|
'input' => 'some//foo',
|
||||||
|
'expected' => 'some/foo',
|
||||||
|
],
|
||||||
|
'three slashes' => [
|
||||||
|
'input' => 'some///foo',
|
||||||
|
'expected' => 'some/foo',
|
||||||
|
],
|
||||||
|
'slashes in front' => [
|
||||||
|
'input' => '/foo',
|
||||||
|
'expected' => '/foo',
|
||||||
|
],
|
||||||
|
'two slashes in front' => [
|
||||||
|
'input' => '//foo',
|
||||||
|
'expected' => '/foo',
|
||||||
|
],
|
||||||
|
'thee slashes in front' => [
|
||||||
|
'input' => '///foo',
|
||||||
|
'expected' => '/foo',
|
||||||
|
],
|
||||||
|
'slashes in back' => [
|
||||||
|
'input' => 'foo/',
|
||||||
|
'expected' => 'foo/',
|
||||||
|
],
|
||||||
|
'two slashes in back' => [
|
||||||
|
'input' => 'foo//',
|
||||||
|
'expected' => 'foo/',
|
||||||
|
],
|
||||||
|
'thee slashes in back' => [
|
||||||
|
'input' => 'foo///',
|
||||||
|
'expected' => 'foo/',
|
||||||
|
],
|
||||||
|
'multiple slashes' => [
|
||||||
|
'input' => '/foo//bar///string/end_times',
|
||||||
|
'expected' => '/foo/bar/string/end_times',
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test multiple slashes clean up
|
||||||
|
*
|
||||||
|
* @covers ::stripMultiplePathSlashes
|
||||||
|
* @dataProvider stripMultiplePathSlashesProvider
|
||||||
|
* @testdox stripMultiplePathSlashes $input will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $input
|
||||||
|
* @param string $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testStripMultiplePathSlashes(string $input, string $expected): void
|
||||||
|
{
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Convert\Strings::stripMultiplePathSlashes($input)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerStripUTF8BomBytes(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
"utf8-bom" => [
|
||||||
|
"file" => "UTF8BOM.csv",
|
||||||
|
"expect" => "Asset Type,Epic,File Name\n",
|
||||||
|
],
|
||||||
|
"utf8" => [
|
||||||
|
"file" => "UTF8.csv",
|
||||||
|
"expect" => "Asset Type,Epic,File Name\n",
|
||||||
|
],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test utf8 bom remove
|
||||||
|
*
|
||||||
|
* @covers ::stripUTF8BomBytes
|
||||||
|
* @dataProvider providerStripUTF8BomBytes
|
||||||
|
* @testdox stripUTF8BomBytes $file will be $expected [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $file
|
||||||
|
* @param string $expected
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testStripUTF8BomBytes(string $file, string $expected): void
|
||||||
|
{
|
||||||
|
// load sample file
|
||||||
|
if (!is_file(self::DATA_FOLDER . $file)) {
|
||||||
|
$this->markTestSkipped('File: ' . $file . ' could not be opened');
|
||||||
|
}
|
||||||
|
$file = file_get_contents(self::DATA_FOLDER . $file);
|
||||||
|
if ($file === false) {
|
||||||
|
$this->markTestSkipped('File: ' . $file . ' could not be read');
|
||||||
|
}
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
\CoreLibs\Convert\Strings::stripUTF8BomBytes($file)
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
1
test/phpunit/Convert/data/UTF8.csv
Normal file
1
test/phpunit/Convert/data/UTF8.csv
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Asset Type,Epic,File Name
|
||||||
|
1
test/phpunit/Convert/data/UTF8BOM.csv
Normal file
1
test/phpunit/Convert/data/UTF8BOM.csv
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Asset Type,Epic,File Name
|
||||||
|
@@ -22,7 +22,6 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
public function sessionProvider(): array
|
public function sessionProvider(): array
|
||||||
{
|
{
|
||||||
// 0: session name as parameter or for GLOBAL value
|
// 0: session name as parameter or for GLOBAL value
|
||||||
// 1: type p: parameter, g: global, d: php.ini default
|
|
||||||
// 2: mock data as array
|
// 2: mock data as array
|
||||||
// checkCliStatus: true/false,
|
// checkCliStatus: true/false,
|
||||||
// getSessionStatus: PHP_SESSION_DISABLED for abort,
|
// getSessionStatus: PHP_SESSION_DISABLED for abort,
|
||||||
@@ -31,13 +30,10 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
// checkActiveSession: true/false, [1st call, 2nd call]
|
// checkActiveSession: true/false, [1st call, 2nd call]
|
||||||
// getSessionId: string or false
|
// getSessionId: string or false
|
||||||
// 3: exepcted name (session)]
|
// 3: exepcted name (session)]
|
||||||
// 4: Exception thrown on error
|
// 4: auto write close flag
|
||||||
// 5: exception code, null for none
|
|
||||||
// 6: expected error string
|
|
||||||
return [
|
return [
|
||||||
'session parameter' => [
|
'session parameter' => [
|
||||||
'sessionNameParameter',
|
'sessionNameParameter',
|
||||||
'p',
|
|
||||||
[
|
[
|
||||||
'checkCliStatus' => false,
|
'checkCliStatus' => false,
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
'getSessionStatus' => PHP_SESSION_NONE,
|
||||||
@@ -47,12 +43,9 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
],
|
],
|
||||||
'sessionNameParameter',
|
'sessionNameParameter',
|
||||||
null,
|
null,
|
||||||
null,
|
|
||||||
'',
|
|
||||||
],
|
],
|
||||||
'session globals' => [
|
'session globals' => [
|
||||||
'sessionNameGlobals',
|
'sessionNameGlobals',
|
||||||
'g',
|
|
||||||
[
|
[
|
||||||
'checkCliStatus' => false,
|
'checkCliStatus' => false,
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
'getSessionStatus' => PHP_SESSION_NONE,
|
||||||
@@ -61,13 +54,10 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
'getSessionId' => '1234abcd4567'
|
'getSessionId' => '1234abcd4567'
|
||||||
],
|
],
|
||||||
'sessionNameGlobals',
|
'sessionNameGlobals',
|
||||||
null,
|
false,
|
||||||
null,
|
|
||||||
'',
|
|
||||||
],
|
],
|
||||||
'session name default' => [
|
'auto write close' => [
|
||||||
'',
|
'sessionNameAutoWriteClose',
|
||||||
'd',
|
|
||||||
[
|
[
|
||||||
'checkCliStatus' => false,
|
'checkCliStatus' => false,
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
'getSessionStatus' => PHP_SESSION_NONE,
|
||||||
@@ -75,109 +65,8 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
'checkActiveSession' => [false, true],
|
'checkActiveSession' => [false, true],
|
||||||
'getSessionId' => '1234abcd4567'
|
'getSessionId' => '1234abcd4567'
|
||||||
],
|
],
|
||||||
'',
|
'sessionNameAutoWriteClose',
|
||||||
null,
|
true,
|
||||||
null,
|
|
||||||
'',
|
|
||||||
],
|
|
||||||
// error checks
|
|
||||||
// 1: we are in cli
|
|
||||||
'on cli error' => [
|
|
||||||
'',
|
|
||||||
'd',
|
|
||||||
[
|
|
||||||
'checkCliStatus' => true,
|
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
|
||||||
'setSessionName' => true,
|
|
||||||
'checkActiveSession' => [false, true],
|
|
||||||
'getSessionId' => '1234abcd4567'
|
|
||||||
],
|
|
||||||
'',
|
|
||||||
'RuntimeException',
|
|
||||||
1,
|
|
||||||
'[SESSION] No sessions in php cli'
|
|
||||||
],
|
|
||||||
// 2: session disabled
|
|
||||||
'session disabled error' => [
|
|
||||||
'',
|
|
||||||
'd',
|
|
||||||
[
|
|
||||||
'checkCliStatus' => false,
|
|
||||||
'getSessionStatus' => PHP_SESSION_DISABLED,
|
|
||||||
'setSessionName' => true,
|
|
||||||
'checkActiveSession' => [false, true],
|
|
||||||
'getSessionId' => '1234abcd4567'
|
|
||||||
],
|
|
||||||
'',
|
|
||||||
'RuntimeException',
|
|
||||||
2,
|
|
||||||
'[SESSION] Sessions are disabled'
|
|
||||||
],
|
|
||||||
// 3: invalid session name: string
|
|
||||||
'invalid name chars error' => [
|
|
||||||
'1invalid$session#;',
|
|
||||||
'p',
|
|
||||||
[
|
|
||||||
'checkCliStatus' => false,
|
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
|
||||||
'setSessionName' => false,
|
|
||||||
'checkActiveSession' => [false, true],
|
|
||||||
'getSessionId' => '1234abcd4567'
|
|
||||||
],
|
|
||||||
'',
|
|
||||||
'UnexpectedValueException',
|
|
||||||
3,
|
|
||||||
'[SESSION] Invalid session name: 1invalid$session#;'
|
|
||||||
],
|
|
||||||
// 3: invalid session name: only numbers
|
|
||||||
'invalid name numbers only error' => [
|
|
||||||
'123',
|
|
||||||
'p',
|
|
||||||
[
|
|
||||||
'checkCliStatus' => false,
|
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
|
||||||
'setSessionName' => false,
|
|
||||||
'checkActiveSession' => [false, true],
|
|
||||||
'getSessionId' => '1234abcd4567'
|
|
||||||
],
|
|
||||||
'',
|
|
||||||
'UnexpectedValueException',
|
|
||||||
3,
|
|
||||||
'[SESSION] Invalid session name: 123'
|
|
||||||
],
|
|
||||||
// 3: invalid session name: invalid name short
|
|
||||||
// 3: invalid session name: too long (128)
|
|
||||||
// 4: failed to start session (2nd false on check active session)
|
|
||||||
'invalid name numbers only error' => [
|
|
||||||
'',
|
|
||||||
'd',
|
|
||||||
[
|
|
||||||
'checkCliStatus' => false,
|
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
|
||||||
'setSessionName' => true,
|
|
||||||
'checkActiveSession' => [false, false],
|
|
||||||
'getSessionId' => '1234abcd4567'
|
|
||||||
],
|
|
||||||
'',
|
|
||||||
'RuntimeException',
|
|
||||||
4,
|
|
||||||
'[SESSION] Failed to activate session'
|
|
||||||
],
|
|
||||||
// 5: get session id return false
|
|
||||||
'invalid name numbers only error' => [
|
|
||||||
'',
|
|
||||||
'd',
|
|
||||||
[
|
|
||||||
'checkCliStatus' => false,
|
|
||||||
'getSessionStatus' => PHP_SESSION_NONE,
|
|
||||||
'setSessionName' => true,
|
|
||||||
'checkActiveSession' => [false, true],
|
|
||||||
'getSessionId' => false
|
|
||||||
],
|
|
||||||
'',
|
|
||||||
'UnexpectedValueException',
|
|
||||||
5,
|
|
||||||
'[SESSION] getSessionId did not return a session id'
|
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -190,32 +79,23 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
* @testdox startSession $input name for $type will be $expected (error: $expected_error) [$_dataName]
|
* @testdox startSession $input name for $type will be $expected (error: $expected_error) [$_dataName]
|
||||||
*
|
*
|
||||||
* @param string $input
|
* @param string $input
|
||||||
* @param string $type
|
|
||||||
* @param array<mixed> $mock_data
|
* @param array<mixed> $mock_data
|
||||||
* @param string $expected
|
* @param string $expected
|
||||||
* @param string|null $exception
|
|
||||||
* @param string $expected_error
|
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testStartSession(
|
public function testStartSession(
|
||||||
string $input,
|
string $input,
|
||||||
string $type,
|
|
||||||
array $mock_data,
|
array $mock_data,
|
||||||
string $expected,
|
string $expected,
|
||||||
?string $exception,
|
?bool $auto_write_close,
|
||||||
?int $exception_code,
|
|
||||||
string $expected_error
|
|
||||||
): void {
|
): void {
|
||||||
// override expected
|
|
||||||
if ($type == 'd') {
|
|
||||||
$expected = ini_get('session.name');
|
|
||||||
}
|
|
||||||
/** @var \CoreLibs\Create\Session&MockObject $session_mock */
|
/** @var \CoreLibs\Create\Session&MockObject $session_mock */
|
||||||
$session_mock = $this->createPartialMock(
|
$session_mock = $this->createPartialMock(
|
||||||
\CoreLibs\Create\Session::class,
|
\CoreLibs\Create\Session::class,
|
||||||
[
|
[
|
||||||
'checkCliStatus', 'getSessionStatus', 'checkActiveSession',
|
'checkCliStatus',
|
||||||
'setSessionName', 'startSessionCall', 'getSessionId',
|
'getSessionStatus', 'checkActiveSession',
|
||||||
|
'getSessionId',
|
||||||
'getSessionName'
|
'getSessionName'
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
@@ -234,12 +114,8 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
$mock_data['checkActiveSession'][0],
|
$mock_data['checkActiveSession'][0],
|
||||||
$mock_data['checkActiveSession'][1],
|
$mock_data['checkActiveSession'][1],
|
||||||
);
|
);
|
||||||
// dummy set for session name
|
|
||||||
$session_mock->method('setSessionName')->with($input)->willReturn($mock_data['setSessionName']);
|
|
||||||
// set session name & return bsed on request data
|
// set session name & return bsed on request data
|
||||||
$session_mock->method('getSessionName')->willReturn($expected);
|
$session_mock->method('getSessionName')->willReturn($expected);
|
||||||
// will not return anything
|
|
||||||
$session_mock->method('startSessionCall');
|
|
||||||
// in test case only return string
|
// in test case only return string
|
||||||
// false: will return false
|
// false: will return false
|
||||||
$session_mock->method('getSessionId')->willReturn($mock_data['getSessionId']);
|
$session_mock->method('getSessionId')->willReturn($mock_data['getSessionId']);
|
||||||
@@ -247,25 +123,7 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
// regex for session id
|
// regex for session id
|
||||||
$ression_id_regex = "/^\w+$/";
|
$ression_id_regex = "/^\w+$/";
|
||||||
|
|
||||||
if ($exception !== null) {
|
$session_id = $session_mock->getSessionId();
|
||||||
$this->expectException($exception);
|
|
||||||
$this->expectExceptionCode($exception_code);
|
|
||||||
}
|
|
||||||
|
|
||||||
unset($GLOBALS['SET_SESSION_NAME']);
|
|
||||||
$session_id = '';
|
|
||||||
switch ($type) {
|
|
||||||
case 'p':
|
|
||||||
$session_id = $session_mock->startSession($input);
|
|
||||||
break;
|
|
||||||
case 'g':
|
|
||||||
$GLOBALS['SET_SESSION_NAME'] = $input;
|
|
||||||
$session_id = $session_mock->startSession();
|
|
||||||
break;
|
|
||||||
case 'd':
|
|
||||||
$session_id = $session_mock->startSession();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
// asert checks
|
// asert checks
|
||||||
if (!empty($session_id)) {
|
if (!empty($session_id)) {
|
||||||
$this->assertMatchesRegularExpression(
|
$this->assertMatchesRegularExpression(
|
||||||
@@ -284,6 +142,73 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerSessionException(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'not cli' => [
|
||||||
|
'TEST_EXCEPTION',
|
||||||
|
\RuntimeException::class,
|
||||||
|
1,
|
||||||
|
'/^\[SESSION\] No sessions in php cli$/',
|
||||||
|
],
|
||||||
|
/* 'session disabled ' => [
|
||||||
|
'TEST_EXCEPTION',
|
||||||
|
\RuntimeException::class,
|
||||||
|
2,
|
||||||
|
'/^\[SESSION\] Sessions are disabled/'
|
||||||
|
],
|
||||||
|
'invalid session name' => [
|
||||||
|
'--#as^-292p-',
|
||||||
|
\UnexpectedValueException::class,
|
||||||
|
3,
|
||||||
|
'/^\[SESSION\] Invalid session name: /'
|
||||||
|
],
|
||||||
|
'failed to activate session' => [
|
||||||
|
'TEST_EXCEPTION',
|
||||||
|
\RuntimeException::class,
|
||||||
|
4,
|
||||||
|
'/^\[SESSION\] Failed to activate session/'
|
||||||
|
],
|
||||||
|
'not a valid session id returned' => [
|
||||||
|
\UnexpectedValueException::class,
|
||||||
|
5,
|
||||||
|
'/^\[SESSION\] getSessionId did not return a session id/'
|
||||||
|
], */
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* exception checks
|
||||||
|
*
|
||||||
|
* @covers ::initSession
|
||||||
|
* @dataProvider providerSessionException
|
||||||
|
* @testdox create session $session_name with exception $exception ($exception_code) [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $session_name
|
||||||
|
* @param string $exception
|
||||||
|
* @param int $exception_code
|
||||||
|
* @param string $expected_error
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testSessionException(
|
||||||
|
string $session_name,
|
||||||
|
string $exception,
|
||||||
|
int $exception_code,
|
||||||
|
string $expected_error,
|
||||||
|
): void {
|
||||||
|
//
|
||||||
|
// throws only on new Object creation
|
||||||
|
$this->expectException($exception);
|
||||||
|
$this->expectExceptionCode($exception_code);
|
||||||
|
$this->expectExceptionMessageMatches($expected_error);
|
||||||
|
new \CoreLibs\Create\Session($session_name);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* provider for session name check
|
* provider for session name check
|
||||||
*
|
*
|
||||||
@@ -347,109 +272,147 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function sessionDataProvider(): array
|
public function providerSessionData(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'test' => [
|
'test' => [
|
||||||
'foo',
|
'foo',
|
||||||
'bar',
|
'bar',
|
||||||
'bar',
|
'bar',
|
||||||
|
null,
|
||||||
],
|
],
|
||||||
'int key test' => [
|
'int key test' => [
|
||||||
123,
|
123,
|
||||||
'bar',
|
'bar',
|
||||||
'bar',
|
'bar',
|
||||||
|
\UnexpectedValueException::class
|
||||||
],
|
],
|
||||||
// more complex value tests
|
// more complex value tests
|
||||||
'array values' => [
|
'array values' => [
|
||||||
'array',
|
'array',
|
||||||
[1, 2, 3],
|
[1, 2, 3],
|
||||||
[1, 2, 3],
|
[1, 2, 3],
|
||||||
|
null,
|
||||||
]
|
]
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NOTE: with auto start session, we cannot test this in the command line
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* method call test
|
* method call test
|
||||||
*
|
*
|
||||||
* @covers ::setS
|
* @covers ::set
|
||||||
* @covers ::getS
|
* @covers ::get
|
||||||
* @covers ::issetS
|
* @covers ::isset
|
||||||
* @covers ::unsetS
|
* @covers ::unset
|
||||||
* @dataProvider sessionDataProvider
|
* @dataProvider providerSessionData
|
||||||
* @testdox setS/getS/issetS/unsetS $name with $input is $expected [$_dataName]
|
* @testdox set/get/isset/unset $name with $input is $expected ($exception) [$_dataName]
|
||||||
*
|
*
|
||||||
* @param string|int $name
|
* @param string|int $name
|
||||||
* @param mixed $input
|
* @param mixed $input
|
||||||
* @param mixed $expected
|
* @param mixed $expected
|
||||||
|
* @param ?mixed $exception
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testMethodSetGet($name, $input, $expected): void
|
public function testMethodSetGet($name, $input, $expected, $exception): void
|
||||||
{
|
{
|
||||||
$session = new \CoreLibs\Create\Session();
|
if (\CoreLibs\Get\System::checkCLI()) {
|
||||||
$session->setS($name, $input);
|
$this->markTestSkipped('Cannot run testMethodSetGet in CLI');
|
||||||
|
}
|
||||||
|
$session = new \CoreLibs\Create\Session('TEST_METHOD');
|
||||||
|
if ($expected !== null) {
|
||||||
|
$this->expectException($exception);
|
||||||
|
}
|
||||||
|
$session->set($name, $input);
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$expected,
|
$expected,
|
||||||
$session->getS($name),
|
$session->get($name),
|
||||||
'method set assert'
|
'method set assert'
|
||||||
);
|
);
|
||||||
// isset true
|
// isset true
|
||||||
$this->assertTrue(
|
$this->assertTrue(
|
||||||
$session->issetS($name),
|
$session->isset($name),
|
||||||
'method isset assert ok'
|
'method isset assert ok'
|
||||||
);
|
);
|
||||||
$session->unsetS($name);
|
$session->unset($name);
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
'',
|
'',
|
||||||
$session->getS($name),
|
$session->get($name),
|
||||||
'method unset assert'
|
'method unset assert'
|
||||||
);
|
);
|
||||||
// iset false
|
// isset false
|
||||||
$this->assertFalse(
|
$this->assertFalse(
|
||||||
$session->issetS($name),
|
$session->isset($name),
|
||||||
'method isset assert false'
|
'method isset assert false'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* magic call test
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @covers ::__set
|
* @return array
|
||||||
* @covers ::__get
|
*/
|
||||||
* @covers ::__isset
|
public function providerSessionDataMany(): array
|
||||||
* @covers ::__unset
|
{
|
||||||
* @dataProvider sessionDataProvider
|
return [
|
||||||
* @testdox __set/__get/__iseet/__unset $name with $input is $expected [$_dataName]
|
'valid set' => [
|
||||||
|
[
|
||||||
|
'foo 1' => 'bar 1',
|
||||||
|
'foo 2' => 'bar 1',
|
||||||
|
],
|
||||||
|
[
|
||||||
|
'foo 1' => 'bar 1',
|
||||||
|
'foo 2' => 'bar 1',
|
||||||
|
],
|
||||||
|
null,
|
||||||
|
],
|
||||||
|
'invalid entry' => [
|
||||||
|
[
|
||||||
|
'foo 1' => 'bar 1',
|
||||||
|
123 => 'bar 1',
|
||||||
|
],
|
||||||
|
[
|
||||||
|
'foo 1' => 'bar 1',
|
||||||
|
],
|
||||||
|
\UnexpectedValueException::class
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
*
|
*
|
||||||
* @param string|int $name
|
* @covers ::setMany
|
||||||
* @param mixed $input
|
* @covers ::getMany
|
||||||
* @param mixed $expected
|
* @dataProvider providerSessionDataMany
|
||||||
|
* @testdox setMany/getMany/unsetMany $set is $expected ($exception) [$_dataName]
|
||||||
|
*
|
||||||
|
* @param array<string|int,mixed> $set
|
||||||
|
* @param array<string,mixed> $expected
|
||||||
|
* @param ?mixed $exception
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function testMagicSetGet($name, $input, $expected): void
|
public function testMany($set, $expected, $exception): void
|
||||||
{
|
{
|
||||||
$session = new \CoreLibs\Create\Session();
|
if (\CoreLibs\Get\System::checkCLI()) {
|
||||||
$session->$name = $input;
|
$this->markTestSkipped('Cannot run testMethodSetGet in CLI');
|
||||||
|
}
|
||||||
|
$session = new \CoreLibs\Create\Session('TEST_METHOD');
|
||||||
|
if ($expected !== null) {
|
||||||
|
$this->expectException($exception);
|
||||||
|
}
|
||||||
|
$session->setMany($set);
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$expected,
|
$expected,
|
||||||
$session->$name,
|
$session->getMany(array_keys($set)),
|
||||||
'magic set assert'
|
'set many failed'
|
||||||
);
|
);
|
||||||
// isset true
|
$session->unsetMany(array_keys($set));
|
||||||
$this->assertTrue(
|
|
||||||
isset($session->$name),
|
|
||||||
'magic isset assert ok'
|
|
||||||
);
|
|
||||||
unset($session->$name);
|
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
'',
|
[],
|
||||||
$session->$name,
|
$session->getMany(array_keys($set)),
|
||||||
'magic unset assert'
|
'unset many failed'
|
||||||
);
|
|
||||||
// isset true
|
|
||||||
$this->assertFalse(
|
|
||||||
isset($session->$name),
|
|
||||||
'magic isset assert false'
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -463,27 +426,30 @@ final class CoreLibsCreateSessionTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testUnsetAll(): void
|
public function testUnsetAll(): void
|
||||||
{
|
{
|
||||||
|
if (\CoreLibs\Get\System::checkCLI()) {
|
||||||
|
$this->markTestSkipped('Cannot run testUnsetAll in CLI');
|
||||||
|
}
|
||||||
$test_values = [
|
$test_values = [
|
||||||
'foo' => 'abc',
|
'foo' => 'abc',
|
||||||
'bar' => '123'
|
'bar' => '123'
|
||||||
];
|
];
|
||||||
$session = new \CoreLibs\Create\Session();
|
$session = new \CoreLibs\Create\Session('TEST_UNSET');
|
||||||
foreach ($test_values as $name => $value) {
|
foreach ($test_values as $name => $value) {
|
||||||
$session->setS($name, $value);
|
$session->set($name, $value);
|
||||||
// confirm set
|
// confirm set
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$value,
|
$value,
|
||||||
$session->getS($name),
|
$session->get($name),
|
||||||
'set assert: ' . $name
|
'set assert: ' . $name
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
// unset all
|
// unset all
|
||||||
$session->unsetAllS();
|
$session->clear();
|
||||||
// check unset
|
// check unset
|
||||||
foreach (array_keys($test_values) as $name) {
|
foreach (array_keys($test_values) as $name) {
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
'',
|
'',
|
||||||
$session->getS($name),
|
$session->get($name),
|
||||||
'unsert assert: ' . $name
|
'unsert assert: ' . $name
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -121,6 +121,7 @@ final class CoreLibsCreateUidsTest extends TestCase
|
|||||||
* must match 7e78fe0d-59b8-4637-af7f-e88d221a7d1e
|
* must match 7e78fe0d-59b8-4637-af7f-e88d221a7d1e
|
||||||
*
|
*
|
||||||
* @covers ::uuidv4
|
* @covers ::uuidv4
|
||||||
|
* @covers ::validateUuidv4
|
||||||
* @testdox uuidv4 check that return is matching regex [$_dataName]
|
* @testdox uuidv4 check that return is matching regex [$_dataName]
|
||||||
*
|
*
|
||||||
* @return void
|
* @return void
|
||||||
@@ -129,13 +130,18 @@ final class CoreLibsCreateUidsTest extends TestCase
|
|||||||
{
|
{
|
||||||
$uuid = \CoreLibs\Create\Uids::uuidv4();
|
$uuid = \CoreLibs\Create\Uids::uuidv4();
|
||||||
$this->assertMatchesRegularExpression(
|
$this->assertMatchesRegularExpression(
|
||||||
'/^[a-z0-9]{8}-[a-z0-9]{4}-[a-z0-9]{4}-[a-z0-9]{4}-[a-z0-9]{12}$/',
|
'/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/',
|
||||||
$uuid
|
$uuid,
|
||||||
|
'Failed regex check'
|
||||||
|
);
|
||||||
|
$this->assertTrue(
|
||||||
|
\CoreLibs\Create\Uids::validateUuuidv4($uuid),
|
||||||
|
'Failed validate regex method'
|
||||||
|
);
|
||||||
|
$this->assertFalse(
|
||||||
|
\CoreLibs\Create\Uids::validateUuuidv4('not-a-uuidv4'),
|
||||||
|
'Failed wrong uuid validated as true'
|
||||||
);
|
);
|
||||||
// $this->assertStringMatchesFormat(
|
|
||||||
// '%4s%4s-%4s-%4s-%4s-%4s%4s%4s',
|
|
||||||
// $uuid
|
|
||||||
// );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ use PHPUnit\Framework\TestCase;
|
|||||||
* Test class for DB\Extended\ArrayIO
|
* Test class for DB\Extended\ArrayIO
|
||||||
* This will only test the PgSQL parts
|
* This will only test the PgSQL parts
|
||||||
* @coversDefaultClass \CoreLibs\DB\Extended\ArrayIO
|
* @coversDefaultClass \CoreLibs\DB\Extended\ArrayIO
|
||||||
* @coversDefaultClass \CoreLibs\DB\Extended\ArrayIO
|
|
||||||
* @testdox \CoreLibs\Extended\ArrayIO method tests for extended DB interface
|
* @testdox \CoreLibs\Extended\ArrayIO method tests for extended DB interface
|
||||||
*/
|
*/
|
||||||
final class CoreLibsDBExtendedArrayIOTest extends TestCase
|
final class CoreLibsDBExtendedArrayIOTest extends TestCase
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ Table with Primary Key: table_with_primary_key
|
|||||||
Table without Primary Key: table_without_primary_key
|
Table without Primary Key: table_without_primary_key
|
||||||
|
|
||||||
Table with primary key has additional row:
|
Table with primary key has additional row:
|
||||||
row_primary_key SERIAL PRIMARY KEY,
|
row_primary_key INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
Each table has the following rows
|
Each table has the following rows
|
||||||
row_int INT,
|
row_int INT,
|
||||||
row_numeric NUMERIC,
|
row_numeric NUMERIC,
|
||||||
@@ -37,8 +37,9 @@ namespace tests;
|
|||||||
|
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
use PHPUnit\Framework\MockObject\MockObject;
|
use PHPUnit\Framework\MockObject\MockObject;
|
||||||
use CoreLibs\Logging\Logger\Level;
|
use CoreLibs\Logging;
|
||||||
use CoreLibs\DB\Options\Convert;
|
use CoreLibs\DB\Options\Convert;
|
||||||
|
use CoreLibs\DB\Support\ConvertPlaceholder;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test class for DB\IO + DB\SQL\PgSQL
|
* Test class for DB\IO + DB\SQL\PgSQL
|
||||||
@@ -117,7 +118,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
// define basic connection set valid and one invalid
|
// define basic connection set valid and one invalid
|
||||||
self::$log = new \CoreLibs\Logging\Logging([
|
self::$log = new Logging\Logging([
|
||||||
// 'log_folder' => __DIR__ . DIRECTORY_SEPARATOR . 'log',
|
// 'log_folder' => __DIR__ . DIRECTORY_SEPARATOR . 'log',
|
||||||
'log_folder' => DIRECTORY_SEPARATOR . 'tmp',
|
'log_folder' => DIRECTORY_SEPARATOR . 'tmp',
|
||||||
'log_file_id' => 'CoreLibs-DB-IO-Test',
|
'log_file_id' => 'CoreLibs-DB-IO-Test',
|
||||||
@@ -161,13 +162,9 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
// primary key name is table + '_id'
|
// primary key name is table + '_id'
|
||||||
<<<SQL
|
<<<SQL
|
||||||
CREATE TABLE table_with_primary_key (
|
CREATE TABLE table_with_primary_key (
|
||||||
table_with_primary_key_id SERIAL PRIMARY KEY,
|
table_with_primary_key_id INT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||||
$base_table
|
$base_table
|
||||||
SQL
|
SQL
|
||||||
/* "CREATE TABLE table_with_primary_key ("
|
|
||||||
// primary key name is table + '_id'
|
|
||||||
. "table_with_primary_key_id SERIAL PRIMARY KEY, "
|
|
||||||
. $base_table */
|
|
||||||
);
|
);
|
||||||
$db->dbExec(
|
$db->dbExec(
|
||||||
<<<SQL
|
<<<SQL
|
||||||
@@ -232,7 +229,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$error,
|
$error,
|
||||||
$last_error,
|
$last_error,
|
||||||
'Assert query warning'
|
'Assert query error'
|
||||||
);
|
);
|
||||||
return [$last_warning, $last_error];
|
return [$last_warning, $last_error];
|
||||||
}
|
}
|
||||||
@@ -251,8 +248,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testDbVersion(): void
|
public function testDbVersion(): void
|
||||||
{
|
{
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -276,8 +271,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testDbVersionNumeric(): void
|
public function testDbVersionNumeric(): void
|
||||||
{
|
{
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -306,8 +299,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testDbVersionInfoParameters(): void
|
public function testDbVersionInfoParameters(): void
|
||||||
{
|
{
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -365,8 +356,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testDbVersionInfo(string $parameter, string $expected): void
|
public function testDbVersionInfo(string $parameter, string $expected): void
|
||||||
{
|
{
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -578,11 +567,11 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
);
|
);
|
||||||
$db->dbClose();
|
$db->dbClose();
|
||||||
// second conenction with log set NOT debug
|
// second conenction with log set NOT debug
|
||||||
$log = new \CoreLibs\Logging\Logging([
|
$log = new Logging\Logging([
|
||||||
// 'log_folder' => __DIR__ . DIRECTORY_SEPARATOR . 'log',
|
// 'log_folder' => __DIR__ . DIRECTORY_SEPARATOR . 'log',
|
||||||
'log_folder' => DIRECTORY_SEPARATOR . 'tmp',
|
'log_folder' => DIRECTORY_SEPARATOR . 'tmp',
|
||||||
'log_file_id' => 'CoreLibs-DB-IO-Test',
|
'log_file_id' => 'CoreLibs-DB-IO-Test',
|
||||||
'log_level' => \CoreLibs\Logging\Logger\Level::Notice,
|
'log_level' => Logging\Logger\Level::Notice,
|
||||||
]);
|
]);
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config[$connection],
|
self::$db_config[$connection],
|
||||||
@@ -1592,8 +1581,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $error,
|
string $error,
|
||||||
bool $run_many_times = false
|
bool $run_many_times = false
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -1832,8 +1819,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $error,
|
string $error,
|
||||||
string $insert_data
|
string $insert_data
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -2002,8 +1987,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $error,
|
string $error,
|
||||||
string $insert_data
|
string $insert_data
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -3069,8 +3052,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $error,
|
string $error,
|
||||||
string $insert_data
|
string $insert_data
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -3309,6 +3290,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'query' => 'INSERT INTO table_with_primary_key (row_int, uid) '
|
'query' => 'INSERT INTO table_with_primary_key (row_int, uid) '
|
||||||
. 'VALUES ($1, $2) RETURNING table_with_primary_key_id',
|
. 'VALUES ($1, $2) RETURNING table_with_primary_key_id',
|
||||||
'returning_id' => true,
|
'returning_id' => true,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// update
|
// update
|
||||||
@@ -3343,6 +3325,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'query' => 'UPDATE table_with_primary_key SET row_int = $1, '
|
'query' => 'UPDATE table_with_primary_key SET row_int = $1, '
|
||||||
. 'row_varchar = $2 WHERE uid = $3',
|
. 'row_varchar = $2 WHERE uid = $3',
|
||||||
'returning_id' => false,
|
'returning_id' => false,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// select
|
// select
|
||||||
@@ -3372,6 +3355,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'count' => 1,
|
'count' => 1,
|
||||||
'query' => 'SELECT row_int, uid FROM table_with_primary_key WHERE uid = $1',
|
'query' => 'SELECT row_int, uid FROM table_with_primary_key WHERE uid = $1',
|
||||||
'returning_id' => false,
|
'returning_id' => false,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// any query but with no parameters
|
// any query but with no parameters
|
||||||
@@ -3404,6 +3388,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'count' => 0,
|
'count' => 0,
|
||||||
'query' => 'SELECT row_int, uid FROM table_with_primary_key',
|
'query' => 'SELECT row_int, uid FROM table_with_primary_key',
|
||||||
'returning_id' => false,
|
'returning_id' => false,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// no statement name (25)
|
// no statement name (25)
|
||||||
@@ -3427,6 +3412,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'count' => 0,
|
'count' => 0,
|
||||||
'query' => '',
|
'query' => '',
|
||||||
'returning_id' => false,
|
'returning_id' => false,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// no query (prepare 11)
|
// no query (prepare 11)
|
||||||
@@ -3451,6 +3437,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'count' => 0,
|
'count' => 0,
|
||||||
'query' => '',
|
'query' => '',
|
||||||
'returning_id' => false,
|
'returning_id' => false,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// no db connection (prepare/execute 16)
|
// no db connection (prepare/execute 16)
|
||||||
@@ -3465,7 +3452,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
$read_query,
|
$read_query,
|
||||||
null,
|
null,
|
||||||
null,
|
null,
|
||||||
//
|
// warning: 20
|
||||||
true, '20', '',
|
true, '20', '',
|
||||||
//
|
//
|
||||||
'result', '', '',
|
'result', '', '',
|
||||||
@@ -3480,6 +3467,33 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'count' => 0,
|
'count' => 0,
|
||||||
'query' => 'SELECT row_int, uid FROM table_with_primary_key',
|
'query' => 'SELECT row_int, uid FROM table_with_primary_key',
|
||||||
'returning_id' => false,
|
'returning_id' => false,
|
||||||
|
'placeholder_converted' => [],
|
||||||
|
],
|
||||||
|
],
|
||||||
|
// prepare with different statement name
|
||||||
|
'prepare query with same statement name, different query' => [
|
||||||
|
'double_error',
|
||||||
|
$read_query,
|
||||||
|
// primary key
|
||||||
|
null,
|
||||||
|
// arguments (none)
|
||||||
|
null,
|
||||||
|
// expected return false, warning: no, error: 26
|
||||||
|
false, '', '26',
|
||||||
|
// return expected, warning, error
|
||||||
|
'', '', '',
|
||||||
|
// dummy query for second prepare with wrong query
|
||||||
|
$read_query . ' WHERE uid = $3',
|
||||||
|
[],
|
||||||
|
//
|
||||||
|
$insert_query,
|
||||||
|
//
|
||||||
|
[
|
||||||
|
'pk_name' => '',
|
||||||
|
'count' => 0,
|
||||||
|
'query' => 'SELECT row_int, uid FROM table_with_primary_key',
|
||||||
|
'returning_id' => false,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// insert wrong data count compared to needed (execute 23)
|
// insert wrong data count compared to needed (execute 23)
|
||||||
@@ -3505,10 +3519,12 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
'query' => 'INSERT INTO table_with_primary_key (row_int, uid) VALUES '
|
'query' => 'INSERT INTO table_with_primary_key (row_int, uid) VALUES '
|
||||||
. '($1, $2) RETURNING table_with_primary_key_id',
|
. '($1, $2) RETURNING table_with_primary_key_id',
|
||||||
'returning_id' => true,
|
'returning_id' => true,
|
||||||
|
'placeholder_converted' => [],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// execute does not return a result (22)
|
// execute does not return a result (22)
|
||||||
// TODO execute does not return a result
|
// TODO execute does not return a result
|
||||||
|
// TODO prepared statement with placeholder params auto convert
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -3554,8 +3570,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $insert_data,
|
string $insert_data,
|
||||||
array $prepare_cursor,
|
array $prepare_cursor,
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -3575,6 +3589,9 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
$db->dbPrepare($stm_name, $query) :
|
$db->dbPrepare($stm_name, $query) :
|
||||||
$db->dbPrepare($stm_name, $query, $pk_name);
|
$db->dbPrepare($stm_name, $query, $pk_name);
|
||||||
}
|
}
|
||||||
|
if ($error_prepare == '26') {
|
||||||
|
$prepare_result = $db->dbPrepare($stm_name, $expected_data_query);
|
||||||
|
}
|
||||||
// if result type, or if forced bool
|
// if result type, or if forced bool
|
||||||
if (is_string($expected_prepare) && $expected_prepare == 'result') {
|
if (is_string($expected_prepare) && $expected_prepare == 'result') {
|
||||||
// if PHP or newer, must be Object PgSql\Result
|
// if PHP or newer, must be Object PgSql\Result
|
||||||
@@ -3597,66 +3614,68 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
|
|
||||||
// for non fail prepare test exec
|
// for non fail prepare test exec
|
||||||
// check test result
|
// check test result
|
||||||
$execute_result = $query_data === null ?
|
if (!$error_prepare) {
|
||||||
$db->dbExecute($stm_name) :
|
$execute_result = $query_data === null ?
|
||||||
$db->dbExecute($stm_name, $query_data);
|
$db->dbExecute($stm_name) :
|
||||||
if ($expected_execute == 'result') {
|
$db->dbExecute($stm_name, $query_data);
|
||||||
// if PHP or newer, must be Object PgSql\Result
|
if ($expected_execute == 'result') {
|
||||||
$this->assertIsObject(
|
// if PHP or newer, must be Object PgSql\Result
|
||||||
$execute_result
|
$this->assertIsObject(
|
||||||
);
|
$execute_result
|
||||||
// also check that this is correct instance type
|
);
|
||||||
$this->assertInstanceOf(
|
// also check that this is correct instance type
|
||||||
'PgSql\Result',
|
$this->assertInstanceOf(
|
||||||
$execute_result
|
'PgSql\Result',
|
||||||
);
|
$execute_result
|
||||||
// if this is an select use dbFetchArray to get data and test
|
);
|
||||||
} else {
|
// if this is an select use dbFetchArray to get data and test
|
||||||
$this->assertEquals(
|
} else {
|
||||||
$expected_execute,
|
$this->assertEquals(
|
||||||
$execute_result
|
$expected_execute,
|
||||||
);
|
$execute_result
|
||||||
}
|
);
|
||||||
// error/warning check
|
}
|
||||||
$this->subAssertErrorTest($db, $warning_execute, $error_execute);
|
// error/warning check
|
||||||
// now check test result if expected return is result
|
$this->subAssertErrorTest($db, $warning_execute, $error_execute);
|
||||||
if (
|
// now check test result if expected return is result
|
||||||
$expected_execute == 'result' &&
|
if (
|
||||||
!empty($expected_data_query)
|
$expected_execute == 'result' &&
|
||||||
) {
|
!empty($expected_data_query)
|
||||||
// $expected_data_query
|
) {
|
||||||
// $expected_data
|
// $expected_data_query
|
||||||
$rows = $db->dbReturnArray($expected_data_query);
|
// $expected_data
|
||||||
$this->assertEquals(
|
$rows = $db->dbReturnArray($expected_data_query);
|
||||||
$expected_data,
|
$this->assertEquals(
|
||||||
$rows
|
$expected_data,
|
||||||
);
|
$rows
|
||||||
}
|
);
|
||||||
if (
|
}
|
||||||
$expected_execute == 'result' &&
|
if (
|
||||||
$execute_result !== false &&
|
$expected_execute == 'result' &&
|
||||||
empty($expected_data_query) &&
|
$execute_result !== false &&
|
||||||
count($expected_data)
|
empty($expected_data_query) &&
|
||||||
) {
|
count($expected_data)
|
||||||
// compare previously read data to compare data
|
) {
|
||||||
$compare_data = [];
|
// compare previously read data to compare data
|
||||||
// read in the query data
|
$compare_data = [];
|
||||||
while (is_array($row = $db->dbFetchArray($execute_result, true))) {
|
// read in the query data
|
||||||
$compare_data[] = $row;
|
while (is_array($row = $db->dbFetchArray($execute_result, true))) {
|
||||||
|
$compare_data[] = $row;
|
||||||
|
}
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected_data,
|
||||||
|
$compare_data
|
||||||
|
);
|
||||||
}
|
}
|
||||||
$this->assertEquals(
|
|
||||||
$expected_data,
|
|
||||||
$compare_data
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// check dbGetPrepareCursorValue
|
// check dbGetPrepareCursorValue
|
||||||
foreach (['pk_name', 'count', 'query', 'returning_id'] as $key) {
|
foreach (['pk_name', 'count', 'query', 'returning_id', 'placeholder_converted'] as $key) {
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$prepare_cursor[$key],
|
$prepare_cursor[$key],
|
||||||
$db->dbGetPrepareCursorValue($stm_name, $key),
|
$db->dbGetPrepareCursorValue($stm_name, $key),
|
||||||
'Prepared cursor: ' . $key . ': failed assertion'
|
'Prepared cursor: ' . $key . ': failed assertion'
|
||||||
);
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// reset all data
|
// reset all data
|
||||||
@@ -3844,8 +3863,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $expected_get_var,
|
string $expected_get_var,
|
||||||
string $expected_get_db
|
string $expected_get_db
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config[$connection],
|
self::$db_config[$connection],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -3910,7 +3927,10 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
// 'main::run::run::run::run::run::run::run::runBare::runTest::testDbErrorHandling::dbSetMaxQueryCall
|
// 'main::run::run::run::run::run::run::run::runBare::runTest::testDbErrorHandling::dbSetMaxQueryCall
|
||||||
'source' => "/^(include::)?main::(run::)+runBare::runTest::testDbErrorHandling::dbSetMaxQueryCall$/",
|
'source' => "/^(include::)?main::(run::)+runBare::runTest::testDbErrorHandling::dbSetMaxQueryCall$/",
|
||||||
'pg_error' => '',
|
'pg_error' => '',
|
||||||
'msg' => '',
|
'message' => '',
|
||||||
|
'context' => [
|
||||||
|
'max_calls' => 0
|
||||||
|
]
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
'trigger warning' => [
|
'trigger warning' => [
|
||||||
@@ -3943,8 +3963,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $error_id,
|
string $error_id,
|
||||||
array $expected_history
|
array $expected_history
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -3970,7 +3988,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
foreach ($expected_history as $key => $value) {
|
foreach ($expected_history as $key => $value) {
|
||||||
// check if starts with / because this is regex (timestamp)
|
// check if starts with / because this is regex (timestamp)
|
||||||
// if (substr($expected_2, 0, 1) == '/) {
|
// if (substr($expected_2, 0, 1) == '/) {
|
||||||
if (strpos($value, '/') === 0) {
|
if (!is_array($value) && strpos($value, '/') === 0) {
|
||||||
// this is regex
|
// this is regex
|
||||||
$this->assertMatchesRegularExpression(
|
$this->assertMatchesRegularExpression(
|
||||||
$value,
|
$value,
|
||||||
@@ -4058,8 +4076,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
bool $expected_set_flag,
|
bool $expected_set_flag,
|
||||||
string $expected_get_encoding
|
string $expected_get_encoding
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config[$connection],
|
self::$db_config[$connection],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -4141,8 +4157,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
?string $encoding_php,
|
?string $encoding_php,
|
||||||
string $text
|
string $text
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config[$connection],
|
self::$db_config[$connection],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -4272,8 +4286,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $table,
|
string $table,
|
||||||
string $primary_key
|
string $primary_key
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -4330,7 +4342,7 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
// NOTE if there are different INSERTS before the primary keys
|
// NOTE if there are different INSERTS before the primary keys
|
||||||
// will not match anymore. Must be updated by hand
|
// will not match anymore. Must be updated by hand
|
||||||
// IMPORTANT: if this is stand alone the primary key will not match and fail
|
// IMPORTANT: if this is stand alone the primary key will not match and fail
|
||||||
$table_with_primary_key_id = 68;
|
$table_with_primary_key_id = 70;
|
||||||
// 0: query + returning
|
// 0: query + returning
|
||||||
// 1: params
|
// 1: params
|
||||||
// 1: pk name for db exec
|
// 1: pk name for db exec
|
||||||
@@ -4530,8 +4542,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
array|string|int|null $expected_ret_ext,
|
array|string|int|null $expected_ret_ext,
|
||||||
array $expected_ret_arr
|
array $expected_ret_arr
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -4875,8 +4885,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
array $expected_col_names,
|
array $expected_col_names,
|
||||||
array $expected_col_types
|
array $expected_col_types
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
@@ -5030,12 +5038,333 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
$db->dbClose();
|
$db->dbClose();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: QUERY PLACEHOLDERS
|
||||||
|
|
||||||
|
// test query placeholder detection for all possible sets
|
||||||
|
// ::dbPrepare
|
||||||
|
|
||||||
|
/**
|
||||||
|
* placeholder sql
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function providerDbCountQueryParams(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'one place holder' => [
|
||||||
|
'query' => 'SELECT row_varchar FROM table_with_primary_key WHERE row_varchar = $1',
|
||||||
|
'count' => 1,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'one place holder, json call' => [
|
||||||
|
'query' => "SELECT row_varchar FROM table_with_primary_key WHERE row_jsonb->>'data' = $1",
|
||||||
|
'count' => 1,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'one place holder, <> compare' => [
|
||||||
|
'query' => "SELECT row_varchar FROM table_with_primary_key WHERE row_varchar <> $1",
|
||||||
|
'count' => 1,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'one place holder, named' => [
|
||||||
|
'query' => "SELECT row_varchar FROM table_with_primary_key WHERE row_varchar <> :row_varchar",
|
||||||
|
'count' => 1,
|
||||||
|
'convert' => true,
|
||||||
|
],
|
||||||
|
'no replacement' => [
|
||||||
|
'query' => "SELECT row_varchar FROM table_with_primary_key WHERE row_varchar = '$1'",
|
||||||
|
'count' => 0,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'insert' => [
|
||||||
|
'query' => "INSERT INTO table_with_primary_key (row_varchar, row_jsonb, row_int) VALUES ($1, $2, $3)",
|
||||||
|
'count' => 3,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'update' => [
|
||||||
|
'query' => "UPDATE table_with_primary_key SET row_varchar = $1, row_jsonb = $2, row_int = $3 WHERE row_numeric = $4",
|
||||||
|
'count' => 4,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'multiple, multline' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
SELECT
|
||||||
|
row_varchar
|
||||||
|
FROM
|
||||||
|
table_with_primary_key
|
||||||
|
WHERE
|
||||||
|
row_varchar = $1 AND row_int = $2
|
||||||
|
AND row_numeric = ANY($3)
|
||||||
|
SQL,
|
||||||
|
'count' => 3,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'two digit numbers' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
INSERT INTO table_with_primary_key (
|
||||||
|
row_int, row_numeric, row_varchar, row_varchar_literal, row_json,
|
||||||
|
row_jsonb, row_bytea, row_timestamp, row_date, row_interval
|
||||||
|
) VALUES (
|
||||||
|
$1, $2, $3, $4, $5,
|
||||||
|
$6, $7, $8, $9, $10
|
||||||
|
)
|
||||||
|
SQL,
|
||||||
|
'count' => 10,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'things in brackets' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
SELECT row_varchar
|
||||||
|
FROM table_with_primary_key
|
||||||
|
WHERE
|
||||||
|
row_varchar = $1 AND
|
||||||
|
(row_int = ANY($2) OR row_int = $3)
|
||||||
|
AND row_varchar_literal = $4
|
||||||
|
SQL,
|
||||||
|
'count' => 4,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'number compare' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
SELECT row_varchar
|
||||||
|
FROM table_with_primary_key
|
||||||
|
WHERE
|
||||||
|
row_int >= $1 OR row_int <= $2 OR
|
||||||
|
row_int > $3 OR row_int < $4
|
||||||
|
OR row_int = $5 OR row_int <> $6
|
||||||
|
SQL,
|
||||||
|
'count' => 6,
|
||||||
|
'convert' => false,
|
||||||
|
],
|
||||||
|
'comments in insert' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
INSERT INTO table_with_primary_key (
|
||||||
|
row_int, row_numeric, row_varchar, row_varchar_literal
|
||||||
|
) VALUES (
|
||||||
|
-- comment 1
|
||||||
|
$1, $2,
|
||||||
|
-- comment 2
|
||||||
|
$3
|
||||||
|
-- comment 3
|
||||||
|
, $4
|
||||||
|
)
|
||||||
|
SQL,
|
||||||
|
'count' => 4,
|
||||||
|
'convert' => false
|
||||||
|
],
|
||||||
|
// Note some are not set
|
||||||
|
'a complete set of possible' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
UPDATE table_with_primary_key SET
|
||||||
|
-- ROW
|
||||||
|
row_varchar = $1
|
||||||
|
WHERE
|
||||||
|
row_varchar = ANY($2) AND row_varchar <> $3
|
||||||
|
AND row_varchar > $4 AND row_varchar < $5
|
||||||
|
AND row_varchar >= $6 AND row_varchar <=$7
|
||||||
|
AND row_jsonb->'a' = $8 AND row_jsonb->>$9 = 'a'
|
||||||
|
AND row_jsonb<@$10 AND row_jsonb@>$11
|
||||||
|
AND row_varchar ^@ $12
|
||||||
|
SQL,
|
||||||
|
'count' => 12,
|
||||||
|
'convert' => false,
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Placeholder check and convert tests
|
||||||
|
*
|
||||||
|
* @covers ::dbPrepare
|
||||||
|
* @covers ::__dbCountQueryParams
|
||||||
|
* @onvers ::convertPlaceholderInQuery
|
||||||
|
* @dataProvider providerDbCountQueryParams
|
||||||
|
* @testdox Query replacement count test [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $query
|
||||||
|
* @param int $count
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testDbCountQueryParams(string $query, int $count, bool $convert): void
|
||||||
|
{
|
||||||
|
$db = new \CoreLibs\DB\IO(
|
||||||
|
self::$db_config['valid'],
|
||||||
|
self::$log
|
||||||
|
);
|
||||||
|
$id = sha1($query);
|
||||||
|
$db->dbSetConvertPlaceholder($convert);
|
||||||
|
$db->dbPrepare($id, $query);
|
||||||
|
// print "\n**\n";
|
||||||
|
// print "PCount: " . $db->dbGetPrepareCursorValue($id, 'count') . "\n";
|
||||||
|
// print "\n**\n";
|
||||||
|
$this->assertEquals(
|
||||||
|
$count,
|
||||||
|
$db->dbGetPrepareCursorValue($id, 'count'),
|
||||||
|
'DB count params'
|
||||||
|
);
|
||||||
|
$placeholder = ConvertPlaceholder::convertPlaceholderInQuery($query, null, 'pg');
|
||||||
|
// print "RES: " . print_r($placeholder, true) . "\n";
|
||||||
|
$this->assertEquals(
|
||||||
|
$count,
|
||||||
|
$placeholder['needed'],
|
||||||
|
'convert params'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* query placeholder convert
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function queryPlaceholderReplaceProvider(): array
|
||||||
|
{
|
||||||
|
// WHERE row_varchar = $1
|
||||||
|
return [
|
||||||
|
'select, no change' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
SELECT row_varchar, row_varchar_literal, row_int, row_date
|
||||||
|
FROM table_with_primary_key
|
||||||
|
SQL,
|
||||||
|
'params' => [],
|
||||||
|
'found' => 0,
|
||||||
|
'expected_query' => '',
|
||||||
|
'expected_params' => [],
|
||||||
|
],
|
||||||
|
'select, params ?' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
SELECT row_varchar, row_varchar_literal, row_int, row_date
|
||||||
|
FROM table_with_primary_key
|
||||||
|
WHERE row_varchar = ?
|
||||||
|
SQL,
|
||||||
|
'params' => ['string a'],
|
||||||
|
'found' => 1,
|
||||||
|
'expected_query' => <<<SQL
|
||||||
|
SELECT row_varchar, row_varchar_literal, row_int, row_date
|
||||||
|
FROM table_with_primary_key
|
||||||
|
WHERE row_varchar = $1
|
||||||
|
SQL,
|
||||||
|
'expected_params' => ['string a'],
|
||||||
|
],
|
||||||
|
'select, params :' => [
|
||||||
|
'query' => <<<SQL
|
||||||
|
SELECT row_varchar, row_varchar_literal, row_int, row_date
|
||||||
|
FROM table_with_primary_key
|
||||||
|
WHERE row_varchar = :row_varchar
|
||||||
|
SQL,
|
||||||
|
'params' => [':row_varchar' => 'string a'],
|
||||||
|
'found' => 1,
|
||||||
|
'expected_query' => <<<SQL
|
||||||
|
SELECT row_varchar, row_varchar_literal, row_int, row_date
|
||||||
|
FROM table_with_primary_key
|
||||||
|
WHERE row_varchar = $1
|
||||||
|
SQL,
|
||||||
|
'expected_params' => ['string a'],
|
||||||
|
],
|
||||||
|
// TODO: test with multiple entries
|
||||||
|
// TODO: test with same entry ($1, $1, :var, :var)
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test query string with placeholders convert
|
||||||
|
*
|
||||||
|
* @dataProvider queryPlaceholderReplaceProvider
|
||||||
|
* @testdox Query replacement test [$_dataName]
|
||||||
|
*
|
||||||
|
* @param string $query
|
||||||
|
* @param array $params
|
||||||
|
* @param string $expected_query
|
||||||
|
* @param array $expected_params
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testQueryPlaceholderReplace(
|
||||||
|
string $query,
|
||||||
|
array $params,
|
||||||
|
int $expected_found,
|
||||||
|
string $expected_query,
|
||||||
|
array $expected_params
|
||||||
|
): void {
|
||||||
|
$db = new \CoreLibs\DB\IO(
|
||||||
|
self::$db_config['valid'],
|
||||||
|
self::$log
|
||||||
|
);
|
||||||
|
$db->dbSetConvertPlaceholder(true);
|
||||||
|
//
|
||||||
|
if ($db->dbCheckQueryForSelect($query)) {
|
||||||
|
$res = $db->dbReturnRowParams($query, $params);
|
||||||
|
$converted = $db->dbGetPlaceholderConverted();
|
||||||
|
} else {
|
||||||
|
$db->dbExecParams($query, $params);
|
||||||
|
$converted = $db->dbGetPlaceholderConverted();
|
||||||
|
}
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected_found,
|
||||||
|
$converted['found'],
|
||||||
|
'Found not equal'
|
||||||
|
);
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected_query,
|
||||||
|
$converted['query'],
|
||||||
|
'Query not equal'
|
||||||
|
);
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected_params,
|
||||||
|
$converted['params'],
|
||||||
|
'Params not equal'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test exception for placeholder convert
|
||||||
|
* -> internally converted to error
|
||||||
|
*
|
||||||
|
* @testdox Query Replace error tests
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testQueryPlaceholderReplaceException(): void
|
||||||
|
{
|
||||||
|
$db = new \CoreLibs\DB\IO(
|
||||||
|
self::$db_config['valid'],
|
||||||
|
self::$log
|
||||||
|
);
|
||||||
|
$db->dbSetConvertPlaceholder(true);
|
||||||
|
$db->dbExecParams(
|
||||||
|
<<<SQL
|
||||||
|
SELECT foo FROM bar
|
||||||
|
WHERE a = ? and b = :bname
|
||||||
|
SQL,
|
||||||
|
['a', 'b']
|
||||||
|
);
|
||||||
|
$this->assertEquals(
|
||||||
|
200,
|
||||||
|
$db->dbGetLastError()
|
||||||
|
);
|
||||||
|
|
||||||
|
// catch unset, for :names
|
||||||
|
$db->dbExecParams(
|
||||||
|
<<<SQL
|
||||||
|
SELECT foo FROM bar
|
||||||
|
WHERE a = :aname and b = :bname
|
||||||
|
SQL,
|
||||||
|
[':foo' => 'a', ':bname' => 'b']
|
||||||
|
);
|
||||||
|
$this->assertEquals(
|
||||||
|
210,
|
||||||
|
$db->dbGetLastError()
|
||||||
|
);
|
||||||
|
|
||||||
|
// TODO: other way around for to pdo
|
||||||
|
}
|
||||||
|
|
||||||
// TODO implement below checks
|
// TODO implement below checks
|
||||||
// - complex write sets
|
// - complex write sets
|
||||||
// dbWriteData, dbWriteDataExt
|
// dbWriteData, dbWriteDataExt
|
||||||
// - data debug
|
// - data debug
|
||||||
// dbDumpData
|
// dbDumpData
|
||||||
|
|
||||||
|
// MARK: ASYNC
|
||||||
|
|
||||||
// ASYNC at the end because it has 1s timeout
|
// ASYNC at the end because it has 1s timeout
|
||||||
// - asynchronous executions
|
// - asynchronous executions
|
||||||
// dbExecAsync, dbCheckAsync
|
// dbExecAsync, dbCheckAsync
|
||||||
@@ -5158,8 +5487,6 @@ final class CoreLibsDBIOTest extends TestCase
|
|||||||
string $warning_final,
|
string $warning_final,
|
||||||
string $error_final
|
string $error_final
|
||||||
): void {
|
): void {
|
||||||
// self::$log->setLogLevelAll('debug', true);
|
|
||||||
// self::$log->setLogLevelAll('print', true);
|
|
||||||
$db = new \CoreLibs\DB\IO(
|
$db = new \CoreLibs\DB\IO(
|
||||||
self::$db_config['valid'],
|
self::$db_config['valid'],
|
||||||
self::$log
|
self::$log
|
||||||
|
|||||||
@@ -513,7 +513,7 @@ final class CoreLibsDebugSupportTest extends TestCase
|
|||||||
public function testGetCallerMethodList(array $expected): void
|
public function testGetCallerMethodList(array $expected): void
|
||||||
{
|
{
|
||||||
$compare = Support::getCallerMethodList();
|
$compare = Support::getCallerMethodList();
|
||||||
// 10: legact
|
// 10: legacy
|
||||||
// 11: direct
|
// 11: direct
|
||||||
// 12: full call
|
// 12: full call
|
||||||
switch (count($compare)) {
|
switch (count($compare)) {
|
||||||
@@ -571,6 +571,31 @@ final class CoreLibsDebugSupportTest extends TestCase
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @cover ::getCallStack
|
||||||
|
* @testdox getCallStack check if it returns data [$_dataName]
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testGetCallStack(): void
|
||||||
|
{
|
||||||
|
$call_stack = Support::getCallStack();
|
||||||
|
// print "Get CALL: " . print_r(Support::getCallStack(), true) . "\n";
|
||||||
|
if ($call_stack < 8) {
|
||||||
|
$this->assertFalse(true, 'getCallStack too low: 8');
|
||||||
|
} else {
|
||||||
|
$this->assertTrue(true, 'getCallSteck ok');
|
||||||
|
}
|
||||||
|
// just test top entry
|
||||||
|
$first = array_shift($call_stack);
|
||||||
|
$this->assertStringEndsWith(
|
||||||
|
':tests\CoreLibsDebugSupportTest->testGetCallStack',
|
||||||
|
$first,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* test the lowest one (one above base)
|
* test the lowest one (one above base)
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -216,6 +216,29 @@ final class CoreLibsGetSystemTest extends TestCase
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Undocumented function
|
||||||
|
*
|
||||||
|
* @covers ::getIpAddresses
|
||||||
|
* @testdox getIpAddresses check
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function testGetIpAddresses()
|
||||||
|
{
|
||||||
|
// response must have "REMOTE_ADDR" entry, others are optional
|
||||||
|
// NOTE: we have no IP addresses on command line
|
||||||
|
$this->assertTrue(
|
||||||
|
true,
|
||||||
|
"We do not have REMOTE_ADDR on command line"
|
||||||
|
);
|
||||||
|
// $this->assertContains(
|
||||||
|
// 'REMOTE_ADDR',
|
||||||
|
// array_keys(\CoreLibs\Get\System::getIpAddresses()),
|
||||||
|
// 'failed REMOTE_ADDR assert'
|
||||||
|
// );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// __END__
|
// __END__
|
||||||
|
|||||||
2
test/phpunit/Language/locale_other/.gitignore
vendored
Normal file
2
test/phpunit/Language/locale_other/.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
*
|
||||||
|
!.gitignore
|
||||||
@@ -39,6 +39,11 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'OK',
|
'str' => 'OK',
|
||||||
'expected' => 'ok',
|
'expected' => 'ok',
|
||||||
],
|
],
|
||||||
|
'success' => [
|
||||||
|
'level' => 'success',
|
||||||
|
'str' => 'SUCCESS',
|
||||||
|
'expected' => 'success',
|
||||||
|
],
|
||||||
'info' => [
|
'info' => [
|
||||||
'level' => 'info',
|
'level' => 'info',
|
||||||
'str' => 'INFO',
|
'str' => 'INFO',
|
||||||
@@ -225,6 +230,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'ERROR MESSAGE',
|
'str' => 'ERROR MESSAGE',
|
||||||
'message' => null,
|
'message' => null,
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<ERROR> ERROR MESSAGE',
|
'expected' => '<ERROR> ERROR MESSAGE',
|
||||||
],
|
],
|
||||||
'error, logged' => [
|
'error, logged' => [
|
||||||
@@ -233,6 +239,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'ERROR MESSAGE',
|
'str' => 'ERROR MESSAGE',
|
||||||
'message' => null,
|
'message' => null,
|
||||||
'log_error' => true,
|
'log_error' => true,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<ERROR> ERROR MESSAGE',
|
'expected' => '<ERROR> ERROR MESSAGE',
|
||||||
],
|
],
|
||||||
'error, logged, message' => [
|
'error, logged, message' => [
|
||||||
@@ -241,14 +248,43 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'ERROR MESSAGE',
|
'str' => 'ERROR MESSAGE',
|
||||||
'message' => 'OTHER ERROR MESSAGE',
|
'message' => 'OTHER ERROR MESSAGE',
|
||||||
'log_error' => true,
|
'log_error' => true,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<ERROR> OTHER ERROR MESSAGE',
|
'expected' => '<ERROR> OTHER ERROR MESSAGE',
|
||||||
],
|
],
|
||||||
|
'warn, not logged' => [
|
||||||
|
'id' => '300',
|
||||||
|
'level' => 'warn',
|
||||||
|
'str' => 'WARNING MESSAGE',
|
||||||
|
'message' => null,
|
||||||
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
|
'expected' => '<WARNING> WARNING MESSAGE',
|
||||||
|
],
|
||||||
|
'warn, logged' => [
|
||||||
|
'id' => '300',
|
||||||
|
'level' => 'warn',
|
||||||
|
'str' => 'WARNING MESSAGE',
|
||||||
|
'message' => null,
|
||||||
|
'log_error' => null,
|
||||||
|
'log_warning' => true,
|
||||||
|
'expected' => '<WARNING> WARNING MESSAGE',
|
||||||
|
],
|
||||||
|
'warn, logged, message' => [
|
||||||
|
'id' => '300',
|
||||||
|
'level' => 'warn',
|
||||||
|
'str' => 'WARNING MESSAGE',
|
||||||
|
'message' => 'OTHER WARNING MESSAGE',
|
||||||
|
'log_error' => null,
|
||||||
|
'log_warning' => true,
|
||||||
|
'expected' => '<WARNING> OTHER WARNING MESSAGE',
|
||||||
|
],
|
||||||
'notice' => [
|
'notice' => [
|
||||||
'id' => '100',
|
'id' => '100',
|
||||||
'level' => 'notice',
|
'level' => 'notice',
|
||||||
'str' => 'NOTICE MESSAGE',
|
'str' => 'NOTICE MESSAGE',
|
||||||
'message' => null,
|
'message' => null,
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<NOTICE> NOTICE MESSAGE',
|
'expected' => '<NOTICE> NOTICE MESSAGE',
|
||||||
],
|
],
|
||||||
'notice, message' => [
|
'notice, message' => [
|
||||||
@@ -257,6 +293,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'NOTICE MESSAGE',
|
'str' => 'NOTICE MESSAGE',
|
||||||
'message' => 'OTHER NOTICE MESSAGE',
|
'message' => 'OTHER NOTICE MESSAGE',
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<NOTICE> OTHER NOTICE MESSAGE',
|
'expected' => '<NOTICE> OTHER NOTICE MESSAGE',
|
||||||
],
|
],
|
||||||
'crash' => [
|
'crash' => [
|
||||||
@@ -265,6 +302,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'CRASH MESSAGE',
|
'str' => 'CRASH MESSAGE',
|
||||||
'message' => null,
|
'message' => null,
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<ALERT> CRASH MESSAGE',
|
'expected' => '<ALERT> CRASH MESSAGE',
|
||||||
],
|
],
|
||||||
'crash, message' => [
|
'crash, message' => [
|
||||||
@@ -273,6 +311,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'CRASH MESSAGE',
|
'str' => 'CRASH MESSAGE',
|
||||||
'message' => 'OTHER CRASH MESSAGE',
|
'message' => 'OTHER CRASH MESSAGE',
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<ALERT> OTHER CRASH MESSAGE',
|
'expected' => '<ALERT> OTHER CRASH MESSAGE',
|
||||||
],
|
],
|
||||||
'abort' => [
|
'abort' => [
|
||||||
@@ -281,6 +320,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'ABORT MESSAGE',
|
'str' => 'ABORT MESSAGE',
|
||||||
'message' => null,
|
'message' => null,
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<CRITICAL> ABORT MESSAGE',
|
'expected' => '<CRITICAL> ABORT MESSAGE',
|
||||||
],
|
],
|
||||||
'abort, message' => [
|
'abort, message' => [
|
||||||
@@ -289,6 +329,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'ABORT MESSAGE',
|
'str' => 'ABORT MESSAGE',
|
||||||
'message' => 'OTHER ABORT MESSAGE',
|
'message' => 'OTHER ABORT MESSAGE',
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<CRITICAL> OTHER ABORT MESSAGE',
|
'expected' => '<CRITICAL> OTHER ABORT MESSAGE',
|
||||||
],
|
],
|
||||||
'unknown' => [
|
'unknown' => [
|
||||||
@@ -297,6 +338,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'WRONG LEVEL MESSAGE',
|
'str' => 'WRONG LEVEL MESSAGE',
|
||||||
'message' => null,
|
'message' => null,
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<EMERGENCY> WRONG LEVEL MESSAGE',
|
'expected' => '<EMERGENCY> WRONG LEVEL MESSAGE',
|
||||||
],
|
],
|
||||||
'unknown, message' => [
|
'unknown, message' => [
|
||||||
@@ -305,6 +347,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
'str' => 'WRONG LEVEL MESSAGE',
|
'str' => 'WRONG LEVEL MESSAGE',
|
||||||
'message' => 'OTHER WRONG LEVEL MESSAGE',
|
'message' => 'OTHER WRONG LEVEL MESSAGE',
|
||||||
'log_error' => null,
|
'log_error' => null,
|
||||||
|
'log_warning' => null,
|
||||||
'expected' => '<EMERGENCY> OTHER WRONG LEVEL MESSAGE',
|
'expected' => '<EMERGENCY> OTHER WRONG LEVEL MESSAGE',
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
@@ -321,6 +364,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
* @param string $str
|
* @param string $str
|
||||||
* @param string|null $message
|
* @param string|null $message
|
||||||
* @param bool|null $log_error
|
* @param bool|null $log_error
|
||||||
|
* @param bool|null $log_warning
|
||||||
* @param string $expected
|
* @param string $expected
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
@@ -330,6 +374,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
string $str,
|
string $str,
|
||||||
?string $message,
|
?string $message,
|
||||||
?bool $log_error,
|
?bool $log_error,
|
||||||
|
?bool $log_warning,
|
||||||
string $expected
|
string $expected
|
||||||
): void {
|
): void {
|
||||||
$log = new \CoreLibs\Logging\Logging([
|
$log = new \CoreLibs\Logging\Logging([
|
||||||
@@ -344,7 +389,8 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
$level,
|
$level,
|
||||||
$str,
|
$str,
|
||||||
message: $message,
|
message: $message,
|
||||||
log_error: $log_error
|
log_error: $log_error,
|
||||||
|
log_warning: $log_warning
|
||||||
);
|
);
|
||||||
$file_content = '';
|
$file_content = '';
|
||||||
if (is_file($log->getLogFolder() . $log->getLogFile())) {
|
if (is_file($log->getLogFolder() . $log->getLogFile())) {
|
||||||
@@ -358,6 +404,11 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
$expected,
|
$expected,
|
||||||
$file_content
|
$file_content
|
||||||
);
|
);
|
||||||
|
} elseif ($level == 'warn' && ($log_warning === null || $log_warning === false)) {
|
||||||
|
$this->assertStringNotContainsString(
|
||||||
|
$expected,
|
||||||
|
$file_content
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
$this->assertStringContainsString(
|
$this->assertStringContainsString(
|
||||||
$expected,
|
$expected,
|
||||||
@@ -377,6 +428,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
* @param string $str
|
* @param string $str
|
||||||
* @param string|null $message
|
* @param string|null $message
|
||||||
* @param bool|null $log_error
|
* @param bool|null $log_error
|
||||||
|
* @param bool|null $log_warning
|
||||||
* @param string $expected
|
* @param string $expected
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
@@ -386,6 +438,7 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
string $str,
|
string $str,
|
||||||
?string $message,
|
?string $message,
|
||||||
?bool $log_error,
|
?bool $log_error,
|
||||||
|
?bool $log_warning,
|
||||||
string $expected
|
string $expected
|
||||||
): void {
|
): void {
|
||||||
$log = new \CoreLibs\Logging\Logging([
|
$log = new \CoreLibs\Logging\Logging([
|
||||||
@@ -400,7 +453,8 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
$level,
|
$level,
|
||||||
$str,
|
$str,
|
||||||
message: $message,
|
message: $message,
|
||||||
log_error: $log_error
|
log_error: $log_error,
|
||||||
|
log_warning: $log_warning
|
||||||
);
|
);
|
||||||
$file_content = '';
|
$file_content = '';
|
||||||
if (is_file($log->getLogFolder() . $log->getLogFile())) {
|
if (is_file($log->getLogFolder() . $log->getLogFile())) {
|
||||||
@@ -414,6 +468,11 @@ final class CoreLibsLoggingErrorMessagesTest extends TestCase
|
|||||||
$expected,
|
$expected,
|
||||||
$file_content
|
$file_content
|
||||||
);
|
);
|
||||||
|
} elseif ($level == 'warn' && $log_warning === false) {
|
||||||
|
$this->assertStringNotContainsString(
|
||||||
|
$expected,
|
||||||
|
$file_content
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
$this->assertStringContainsString(
|
$this->assertStringContainsString(
|
||||||
$expected,
|
$expected,
|
||||||
|
|||||||
@@ -46,12 +46,34 @@ final class CoreLibsSecuritySymmetricEncryptionTest extends TestCase
|
|||||||
public function testEncryptDecryptSuccess(string $input, string $expected): void
|
public function testEncryptDecryptSuccess(string $input, string $expected): void
|
||||||
{
|
{
|
||||||
$key = CreateKey::generateRandomKey();
|
$key = CreateKey::generateRandomKey();
|
||||||
$encrypted = SymmetricEncryption::encrypt($input, $key);
|
|
||||||
$decrypted = SymmetricEncryption::decrypt($encrypted, $key);
|
// test class
|
||||||
|
$crypt = new SymmetricEncryption($key);
|
||||||
|
$encrypted = $crypt->encrypt($input);
|
||||||
|
$decrypted = $crypt->decrypt($encrypted);
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
$decrypted,
|
||||||
|
'Class call',
|
||||||
|
);
|
||||||
|
|
||||||
|
// test indirect
|
||||||
|
$encrypted = SymmetricEncryption::getInstance($key)->encrypt($input);
|
||||||
|
$decrypted = SymmetricEncryption::getInstance($key)->decrypt($encrypted);
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
$decrypted,
|
||||||
|
'Class Instance call',
|
||||||
|
);
|
||||||
|
|
||||||
|
// test static
|
||||||
|
$encrypted = SymmetricEncryption::encryptKey($input, $key);
|
||||||
|
$decrypted = SymmetricEncryption::decryptKey($encrypted, $key);
|
||||||
|
|
||||||
$this->assertEquals(
|
$this->assertEquals(
|
||||||
$expected,
|
$expected,
|
||||||
$decrypted
|
$decrypted,
|
||||||
|
'Static call',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -86,10 +108,24 @@ final class CoreLibsSecuritySymmetricEncryptionTest extends TestCase
|
|||||||
public function testEncryptFailed(string $input, string $exception_message): void
|
public function testEncryptFailed(string $input, string $exception_message): void
|
||||||
{
|
{
|
||||||
$key = CreateKey::generateRandomKey();
|
$key = CreateKey::generateRandomKey();
|
||||||
$encrypted = SymmetricEncryption::encrypt($input, $key);
|
|
||||||
$wrong_key = CreateKey::generateRandomKey();
|
$wrong_key = CreateKey::generateRandomKey();
|
||||||
|
|
||||||
|
// wrong key in class call
|
||||||
|
$crypt = new SymmetricEncryption($key);
|
||||||
|
$encrypted = $crypt->encrypt($input);
|
||||||
$this->expectExceptionMessage($exception_message);
|
$this->expectExceptionMessage($exception_message);
|
||||||
SymmetricEncryption::decrypt($encrypted, $wrong_key);
|
$crypt->setKey($key);
|
||||||
|
$crypt->decrypt($encrypted);
|
||||||
|
|
||||||
|
// class instance
|
||||||
|
$encrypted = SymmetricEncryption::getInstance($key)->encrypt($input);
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::getInstance($wrong_key)->decrypt($encrypted);
|
||||||
|
|
||||||
|
// class static
|
||||||
|
$encrypted = SymmetricEncryption::encryptKey($input, $key);
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::decryptKey($encrypted, $wrong_key);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -107,7 +143,6 @@ final class CoreLibsSecuritySymmetricEncryptionTest extends TestCase
|
|||||||
'too short hex key' => [
|
'too short hex key' => [
|
||||||
'key' => '1cabd5cba9e042f12522f4ff2de5c31d233b',
|
'key' => '1cabd5cba9e042f12522f4ff2de5c31d233b',
|
||||||
'excpetion_message' => 'Key is not the correct size (must be '
|
'excpetion_message' => 'Key is not the correct size (must be '
|
||||||
. 'SODIUM_CRYPTO_SECRETBOX_KEYBYTES bytes long).'
|
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -126,13 +161,33 @@ final class CoreLibsSecuritySymmetricEncryptionTest extends TestCase
|
|||||||
*/
|
*/
|
||||||
public function testWrongKey(string $key, string $exception_message): void
|
public function testWrongKey(string $key, string $exception_message): void
|
||||||
{
|
{
|
||||||
$this->expectExceptionMessage($exception_message);
|
|
||||||
SymmetricEncryption::encrypt('test', $key);
|
|
||||||
// we must encrypt valid thing first so we can fail with the wrong kjey
|
|
||||||
$enc_key = CreateKey::generateRandomKey();
|
$enc_key = CreateKey::generateRandomKey();
|
||||||
$encrypted = SymmetricEncryption::encrypt('test', $enc_key);
|
|
||||||
|
// class
|
||||||
|
$crypt = new SymmetricEncryption($key);
|
||||||
$this->expectExceptionMessage($exception_message);
|
$this->expectExceptionMessage($exception_message);
|
||||||
SymmetricEncryption::decrypt($encrypted, $key);
|
$crypt->encrypt('test');
|
||||||
|
$crypt->setKey($enc_key);
|
||||||
|
$encrypted = $crypt->encrypt('test');
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
$crypt->setKey($key);
|
||||||
|
$crypt->decrypt($encrypted);
|
||||||
|
|
||||||
|
// class instance
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::getInstance($key)->encrypt('test');
|
||||||
|
// we must encrypt valid thing first so we can fail with the wrong key
|
||||||
|
$encrypted = SymmetricEncryption::getInstance($enc_key)->encrypt('test');
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::getInstance($key)->decrypt($encrypted);
|
||||||
|
|
||||||
|
// class static
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::encryptKey('test', $key);
|
||||||
|
// we must encrypt valid thing first so we can fail with the wrong key
|
||||||
|
$encrypted = SymmetricEncryption::encryptKey('test', $enc_key);
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::decryptKey($encrypted, $key);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -145,7 +200,7 @@ final class CoreLibsSecuritySymmetricEncryptionTest extends TestCase
|
|||||||
return [
|
return [
|
||||||
'too short ciphertext' => [
|
'too short ciphertext' => [
|
||||||
'input' => 'short',
|
'input' => 'short',
|
||||||
'exception_message' => 'Invalid ciphertext (too short)'
|
'exception_message' => 'Decipher message failed: '
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -164,8 +219,18 @@ final class CoreLibsSecuritySymmetricEncryptionTest extends TestCase
|
|||||||
public function testWrongCiphertext(string $input, string $exception_message): void
|
public function testWrongCiphertext(string $input, string $exception_message): void
|
||||||
{
|
{
|
||||||
$key = CreateKey::generateRandomKey();
|
$key = CreateKey::generateRandomKey();
|
||||||
|
// class
|
||||||
|
$crypt = new SymmetricEncryption($key);
|
||||||
$this->expectExceptionMessage($exception_message);
|
$this->expectExceptionMessage($exception_message);
|
||||||
SymmetricEncryption::decrypt($input, $key);
|
$crypt->decrypt($input);
|
||||||
|
|
||||||
|
// class instance
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::getInstance($key)->decrypt($input);
|
||||||
|
|
||||||
|
// class static
|
||||||
|
$this->expectExceptionMessage($exception_message);
|
||||||
|
SymmetricEncryption::decryptKey($input, $key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
1232
test/phpunit/UrlRequests/CoreLibsUrlRequestsCurlTest.php
Normal file
1232
test/phpunit/UrlRequests/CoreLibsUrlRequestsCurlTest.php
Normal file
File diff suppressed because it is too large
Load Diff
2
tmp/.gitignore
vendored
Normal file
2
tmp/.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
*
|
||||||
|
!.gitignore
|
||||||
@@ -1 +0,0 @@
|
|||||||
/home/clemens/.phive/phars/phan-5.4.2.phar
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user